Last active 2 hours ago

🚀 Автоматизирано изграждане на защитен хостинг с Cloudflare Tunnel & Docker и Wordpress сайтове

setup_hosting.sh Raw
1#!/bin/bash
2# =============================================================================
3# Hosting setup with Cloudflare Tunnel
4# Author Fedya Serafiev
5# Site itpraktika.com
6# =============================================================================
7
8set -euo pipefail
9
10# -- Colors -------------------------------------------------------------------
11RED='\033[0;31m'; GREEN='\033[0;32m'; YELLOW='\033[1;33m'
12CYAN='\033[0;36m'; BOLD='\033[1m'; RESET='\033[0m'
13
14info() { echo -e "${CYAN}[INFO] $*${RESET}"; }
15success() { echo -e "${GREEN}[OK] $*${RESET}"; }
16warn() { echo -e "${YELLOW}[WARN] $*${RESET}"; }
17error() { echo -e "${RED}[ERROR] $*${RESET}" >&2; exit 1; }
18
19# -- Password generator -------------------------------------------------------
20gen_pass() {
21 openssl rand -base64 32 | tr -dc 'a-zA-Z0-9' | head -c 24
22}
23
24# -- Input validation ---------------------------------------------------------
25validate_yn() {
26 [[ "$1" =~ ^[yYnN]$ ]] || error "Vuvedete 'y' ili 'n'."
27}
28
29validate_count() {
30 [[ "$1" =~ ^[0-2]$ ]] || error "Vuvedete 0, 1 ili 2."
31}
32
33# =============================================================================
34echo -e "\n${BOLD}================================================${RESET}"
35echo -e "${BOLD} Nastrojka na Hosting s Cloudflare Tunnel ${RESET}"
36echo -e "${BOLD}================================================${RESET}\n"
37
38# -- [1/4] Cloudflare Tunnel method -------------------------------------------
39echo -e "${BOLD}[1/4] Cloudflare Tunnel - metod na svarzavane${RESET}"
40echo " 1) Token - vuvedash gotov token ot Cloudflare Dashboard"
41echo " 2) Login - cloudflared se logva sam (URL / brauzar)"
42echo ""
43read -rp "Izberi metod [1/2]: " cf_method
44
45case "$cf_method" in
46 1)
47 read -rp " Vuvedete Tunnel Token: " cf_token
48 [[ -n "$cf_token" ]] || error "Tokenat ne mozhe da e prazen."
49 CLOUDFLARED_CMD='tunnel --no-autoupdate run --token ${TUNNEL_TOKEN}'
50 CF_ENV_LINE=" - TUNNEL_TOKEN=\${TUNNEL_TOKEN}"
51 CF_TOKEN_IN_ENV="TUNNEL_TOKEN=$cf_token"
52 ;;
53 2)
54 info "Pri startиrane cloudflared shte izvede URL za avtentikaciya."
55 CLOUDFLARED_CMD='tunnel --no-autoupdate login'
56 CF_ENV_LINE=""
57 CF_TOKEN_IN_ENV="# TUNNEL_TOKEN= (login method)"
58 cf_token=""
59 ;;
60 *)
61 error "Nevaliден izbor. Vuvedete 1 ili 2."
62 ;;
63esac
64
65# -- [2/4] Uptime Kuma --------------------------------------------------------
66echo -e "\n${BOLD}[2/4] Uptime Kuma${RESET}"
67read -rp "Instaliraj Uptime Kuma? (y/n): " install_kuma
68validate_yn "$install_kuma"
69install_kuma="${install_kuma,,}"
70
71# -- [3/4] WordPress sites ----------------------------------------------------
72echo -e "\n${BOLD}[3/4] WordPress sajtove${RESET}"
73read -rp "Broy WordPress sajtove? (0/1/2): " wp_count
74validate_count "$wp_count"
75
76# -- [4/4] Generate passwords -------------------------------------------------
77echo -e "\n${BOLD}[4/4] Generirane na paroli...${RESET}"
78DB_ROOT_PASS=$(gen_pass)
79WP1_DB_PASS=$(gen_pass)
80WP2_DB_PASS=$(gen_pass)
81
82# =============================================================================
83# .env file
84# =============================================================================
85cat > .env <<ENVEOF
86# -- Cloudflare ---------------------------------------------------------------
87$CF_TOKEN_IN_ENV
88
89# -- Database -----------------------------------------------------------------
90DB_ROOT_PASSWORD=$DB_ROOT_PASS
91WP1_DB_PASSWORD=$WP1_DB_PASS
92WP2_DB_PASSWORD=$WP2_DB_PASS
93ENVEOF
94
95chmod 600 .env
96success ".env created (chmod 600)"
97
98# =============================================================================
99# DB init SQL
100# =============================================================================
101mkdir -p db-init
102
103SQL_CONTENT=""
104if [[ "$wp_count" -ge 1 ]]; then
105 SQL_CONTENT+="CREATE DATABASE IF NOT EXISTS wp1_db CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;\n"
106 SQL_CONTENT+="CREATE USER IF NOT EXISTS 'wp1_user'@'%' IDENTIFIED BY '${WP1_DB_PASS}';\n"
107 SQL_CONTENT+="GRANT ALL PRIVILEGES ON wp1_db.* TO 'wp1_user'@'%';\n"
108fi
109if [[ "$wp_count" -ge 2 ]]; then
110 SQL_CONTENT+="CREATE DATABASE IF NOT EXISTS wp2_db CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;\n"
111 SQL_CONTENT+="CREATE USER IF NOT EXISTS 'wp2_user'@'%' IDENTIFIED BY '${WP2_DB_PASS}';\n"
112 SQL_CONTENT+="GRANT ALL PRIVILEGES ON wp2_db.* TO 'wp2_user'@'%';\n"
113fi
114SQL_CONTENT+="FLUSH PRIVILEGES;\n"
115
116printf "%b" "$SQL_CONTENT" > db-init/01-init.sql
117success "db-init/01-init.sql created"
118
119# =============================================================================
120# docker-compose.yml
121# =============================================================================
122cat > docker-compose.yml <<DCEOF
123services:
124
125 # -- Cloudflare Tunnel -------------------------------------------------------
126 cloudflared:
127 image: cloudflare/cloudflared:latest
128 restart: always
129 command: $CLOUDFLARED_CMD
130 networks:
131 - internal
132DCEOF
133
134# Add environment block only for token method
135if [[ -n "$CF_ENV_LINE" ]]; then
136cat >> docker-compose.yml <<DCEOF
137 environment:
138$CF_ENV_LINE
139DCEOF
140fi
141
142# -- MariaDB ------------------------------------------------------------------
143cat >> docker-compose.yml <<DCEOF
144
145 # -- Database ----------------------------------------------------------------
146 db:
147 image: mariadb:10.6
148 restart: always
149 environment:
150 MYSQL_ROOT_PASSWORD: \${DB_ROOT_PASSWORD}
151 volumes:
152 - db_data:/var/lib/mysql
153 - ./db-init:/docker-entrypoint-initdb.d:ro
154 networks:
155 - internal
156 healthcheck:
157 test: ["CMD", "healthcheck.sh", "--connect", "--innodb_initialized"]
158 interval: 10s
159 timeout: 5s
160 retries: 5
161DCEOF
162
163# -- Uptime Kuma --------------------------------------------------------------
164if [[ "$install_kuma" == "y" ]]; then
165cat >> docker-compose.yml <<DCEOF
166
167 # -- Uptime Kuma -------------------------------------------------------------
168 uptime-kuma:
169 image: louislam/uptime-kuma:latest
170 container_name: uptime-kuma
171 restart: always
172 volumes:
173 - kuma_data:/app/data
174 networks:
175 - internal
176DCEOF
177fi
178
179# -- WordPress 1 --------------------------------------------------------------
180if [[ "$wp_count" -ge 1 ]]; then
181cat >> docker-compose.yml <<DCEOF
182
183 # -- WordPress 1 -------------------------------------------------------------
184 wp1:
185 image: wordpress:latest
186 restart: always
187 depends_on:
188 db:
189 condition: service_healthy
190 environment:
191 WORDPRESS_DB_HOST: db
192 WORDPRESS_DB_USER: wp1_user
193 WORDPRESS_DB_PASSWORD: \${WP1_DB_PASSWORD}
194 WORDPRESS_DB_NAME: wp1_db
195 volumes:
196 - wp1_data:/var/www/html
197 networks:
198 - internal
199DCEOF
200fi
201
202# -- WordPress 2 --------------------------------------------------------------
203if [[ "$wp_count" -ge 2 ]]; then
204cat >> docker-compose.yml <<DCEOF
205
206 # -- WordPress 2 -------------------------------------------------------------
207 wp2:
208 image: wordpress:latest
209 restart: always
210 depends_on:
211 db:
212 condition: service_healthy
213 environment:
214 WORDPRESS_DB_HOST: db
215 WORDPRESS_DB_USER: wp2_user
216 WORDPRESS_DB_PASSWORD: \${WP2_DB_PASSWORD}
217 WORDPRESS_DB_NAME: wp2_db
218 volumes:
219 - wp2_data:/var/www/html
220 networks:
221 - internal
222DCEOF
223fi
224
225# -- Networks & Volumes -------------------------------------------------------
226cat >> docker-compose.yml <<DCEOF
227
228networks:
229 internal:
230 driver: bridge
231
232volumes:
233 db_data:
234DCEOF
235
236[[ "$install_kuma" == "y" ]] && echo " kuma_data:" >> docker-compose.yml
237[[ "$wp_count" -ge 1 ]] && echo " wp1_data:" >> docker-compose.yml
238[[ "$wp_count" -ge 2 ]] && echo " wp2_data:" >> docker-compose.yml
239
240success "docker-compose.yml created"
241
242# =============================================================================
243# Summary
244# =============================================================================
245echo ""
246echo -e "${BOLD}================================================${RESET}"
247echo -e "${BOLD} Summary / Rezultat ${RESET}"
248echo -e "${BOLD}================================================${RESET}"
249echo -e " CF method : ${CYAN}$([ "$cf_method" == "1" ] && echo "Token" || echo "Login")${RESET}"
250echo -e " Uptime Kuma : ${CYAN}$([ "$install_kuma" == "y" ] && echo "Yes" || echo "No")${RESET}"
251echo -e " WordPress : ${CYAN}$wp_count site(s)${RESET}"
252echo ""
253echo -e "${BOLD} Files created:${RESET}"
254echo " .env (chmod 600)"
255echo " docker-compose.yml"
256echo " db-init/01-init.sql"
257echo ""
258echo -e "${BOLD} Next step:${RESET}"
259echo -e " ${GREEN}docker compose up -d${RESET}"
260echo ""
261warn "Keep .env in a safe place - it contains your passwords!"
262echo -e "${BOLD}================================================${RESET}\n"