#!/usr/bin/env bash # # This is a Shell script for configure and start WireGuard VPN server. # # Copyright (C) 2019 - 2020 Teddysun # # Reference URL: # https://www.wireguard.com # https://git.zx2c4.com/WireGuard # https://teddysun.com/554.html trap _exit INT QUIT TERM cur_dir="$(cd -P -- "$(dirname -- "$0")" && pwd -P)" [ ${EUID} -ne 0 ] && _red "This script must be run as root\n" && exit 1 _red() { printf '\033[1;31;31m%b\033[0m' "$1" } _green() { printf '\033[1;31;32m%b\033[0m' "$1" } _yellow() { printf '\033[1;31;33m%b\033[0m' "$1" } _printargs() { printf -- "%s" "[$(date)] " printf -- "%s" "$1" printf "\n" } _info() { _printargs "$@" } _warn() { printf -- "%s" "[$(date)] " _yellow "$1" printf "\n" } _error() { printf -- "%s" "[$(date)] " _red "$1" printf "\n" exit 2 } _exit() { printf "\n" _red "$0 has been terminated." printf "\n" exit 1 } _exists() { local cmd="$1" if eval type type > /dev/null 2>&1; then eval type "$cmd" > /dev/null 2>&1 elif command > /dev/null 2>&1; then command -v "$cmd" > /dev/null 2>&1 else which "$cmd" > /dev/null 2>&1 fi local rt=$? return ${rt} } _ipv4() { local ipv4="$( ip addr | egrep -o '[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}' | \ egrep -v "^192\.168|^172\.1[6-9]\.|^172\.2[0-9]\.|^172\.3[0-2]\.|^10\.|^127\.|^255\.|^0\.|^169\.254\." | head -n 1 )" [ -z "${ipv4}" ] && ipv4="$( wget -qO- -t1 -T2 ipv4.icanhazip.com )" [ -z "${ipv4}" ] && ipv4="$( wget -qO- -t1 -T2 ipinfo.io/ip )" printf -- "%s" "${ipv4}" } _ipv6() { local ipv6="" ipv6="$(wget -qO- -t1 -T2 ipv6.icanhazip.com)" printf -- "%s" "${ipv6}" } _nic() { local nic="" nic="$(ip -4 route ls | grep default | grep -Po '(?<=dev )(\S+)' | head -1)" printf -- "%s" "${nic}" } _port() { local port="$(shuf -i 1024-20480 -n 1)" while true do if _exists "netstat" && netstat -tunlp | grep -w "${port}" > /dev/null 2>&1; then port="$(shuf -i 1024-20480 -n 1)" else break fi done printf -- "%s" "${port}" } _os() { local os="" [ -f "/etc/debian_version" ] && source /etc/os-release && os="${ID}" && printf -- "%s" "${os}" && return [ -f "/etc/fedora-release" ] && os="fedora" && printf -- "%s" "${os}" && return [ -f "/etc/redhat-release" ] && os="centos" && printf -- "%s" "${os}" && return } _os_full() { [ -f /etc/redhat-release ] && awk '{print ($1,$3~/^[0-9]/?$3:$4)}' /etc/redhat-release && return [ -f /etc/os-release ] && awk -F'[= "]' '/PRETTY_NAME/{print $3,$4,$5}' /etc/os-release && return [ -f /etc/lsb-release ] && awk -F'[="]+' '/DESCRIPTION/{print $2}' /etc/lsb-release && return } _os_ver() { local main_ver="$( echo $(_os_full) | grep -oE "[0-9.]+")" printf -- "%s" "${main_ver%%.*}" } _error_detect() { local cmd="$1" _info "${cmd}" eval ${cmd} 1> /dev/null if [ $? -ne 0 ]; then _error "Execution command (${cmd}) failed, please check it and try again." fi } _version_gt(){ test "$(echo "$@" | tr " " "\n" | sort -V | head -n 1)" != "$1" } _version_ge(){ test "$(echo "$@" | tr " " "\n" | sort -rV | head -n 1)" == "$1" } _is_installed() { install_flag=(0 0) if _exists "wg" && _exists "wg-quick"; then install_flag[0]=1 fi if [ -s "/lib/modules/$(uname -r)/extra/wireguard.ko" ] \ || [ -s "/lib/modules/$(uname -r)/extra/wireguard.ko.xz" ] \ || [ -s "/lib/modules/$(uname -r)/updates/dkms/wireguard.ko" ] \ || [ -s "/lib/modules/$(uname -r)/kernel/drivers/net/wireguard/wireguard.ko" ] \ || [ -s "/lib/modules/$(uname -r)/kernel/drivers/net/wireguard/wireguard.ko.xz" ]; then install_flag[1]=1 fi if [ "${install_flag[0]}" = "1" ] && [ "${install_flag[1]}" = "1" ]; then return 0 fi if [ "${install_flag[0]}" = "1" ] && [ "${install_flag[1]}" = "0" ]; then return 1 fi if [ "${install_flag[0]}" = "0" ] && [ "${install_flag[1]}" = "1" ]; then return 2 fi if [ "${install_flag[0]}" = "0" ] && [ "${install_flag[1]}" = "0" ]; then return 3 fi } get_latest_module_ver() { wireguard_ver="$(wget --no-check-certificate -qO- https://api.github.com/repos/WireGuard/wireguard-linux-compat/tags | grep 'name' | head -1 | cut -d\" -f4)" if [ -z "${wireguard_ver}" ]; then wireguard_ver="$(curl -Lso- https://api.github.com/repos/WireGuard/wireguard-linux-compat/tags | grep 'name' | head -1 | cut -d\" -f4)" fi if [ -z "${wireguard_ver}" ]; then _error "Failed to get latest wireguard module version from github" fi } get_latest_tools_ver() { wireguard_tools_ver="$(wget --no-check-certificate -qO- https://api.github.com/repos/WireGuard/wireguard-tools/tags | grep 'name' | head -1 | cut -d\" -f4)" if [ -z "${wireguard_tools_ver}" ]; then wireguard_tools_ver="$(curl -Lso- https://api.github.com/repos/WireGuard/wireguard-tools/tags | grep 'name' | head -1 | cut -d\" -f4)" fi if [ -z "${wireguard_tools_ver}" ]; then _error "Failed to get latest wireguard tools version from github" fi } # Check OS version check_os() { _info "Check OS version" if _exists "virt-what"; then virt="$(virt-what)" elif _exists "systemd-detect-virt"; then virt="$(systemd-detect-virt)" fi if [ -n "${virt}" -a "${virt}" = "lxc" ]; then _error "Virtualization method is LXC, which is not supported." fi if [ -n "${virt}" -a "${virt}" = "openvz" ] || [ -d "/proc/vz" ]; then _error "Virtualization method is OpenVZ, which is not supported." fi [ -z "$(_os)" ] && _error "Not supported OS" case "$(_os)" in ubuntu) [ -n "$(_os_ver)" -a "$(_os_ver)" -lt 16 ] && _error "Not supported OS, please change to Ubuntu 16+ and try again." ;; debian|raspbian) [ -n "$(_os_ver)" -a "$(_os_ver)" -lt 8 ] && _error "Not supported OS, please change to De(Rasp)bian 8+ and try again." ;; fedora) [ -n "$(_os_ver)" -a "$(_os_ver)" -lt 29 ] && _error "Not supported OS, please change to Fedora 29+ and try again." ;; centos) [ -n "$(_os_ver)" -a "$(_os_ver)" -lt 7 ] && _error "Not supported OS, please change to CentOS 7+ and try again." ;; *) _error "Not supported OS" ;; esac } # Check linux kernel version check_kernel_version() { kernel_version="$(uname -r | cut -d- -f1)" if _version_ge ${kernel_version} 5.6.0; then return 0 else return 1 fi } # Install wireguard module from source install_wg_module() { get_latest_module_ver wireguard_name="wireguard-linux-compat-$(echo ${wireguard_ver} | grep -oE '[0-9.]+')" wireguard_url="https://github.com/WireGuard/wireguard-linux-compat/archive/${wireguard_ver}.tar.gz" cd ${cur_dir} _error_detect "wget --no-check-certificate -qO ${wireguard_name}.tar.gz ${wireguard_url}" _error_detect "tar zxf ${wireguard_name}.tar.gz" _error_detect "cd ${wireguard_name}/src" _error_detect "make" _error_detect "make install" _error_detect "cd ${cur_dir} && rm -fr ${wireguard_name}.tar.gz ${wireguard_name}" } # Install wireguard tools from source install_wg_tools() { get_latest_tools_ver wireguard_tools_name="wireguard-tools-$(echo ${wireguard_tools_ver} | grep -oE '[0-9.]+')" wireguard_tools_url="https://github.com/WireGuard/wireguard-tools/archive/${wireguard_tools_ver}.tar.gz" cd ${cur_dir} _error_detect "wget --no-check-certificate -qO ${wireguard_tools_name}.tar.gz ${wireguard_tools_url}" _error_detect "tar zxf ${wireguard_tools_name}.tar.gz" _error_detect "cd ${wireguard_tools_name}/src" _error_detect "make" _error_detect "make install" _error_detect "cd ${cur_dir} && rm -fr ${wireguard_tools_name}.tar.gz ${wireguard_tools_name}" } install_wg_pkgs() { _info "Install dependencies for wireguard" case "$(_os)" in ubuntu|debian|raspbian) _error_detect "apt-get update" _error_detect "apt-get -y install qrencode" _error_detect "apt-get -y install iptables" _error_detect "apt-get -y install bc" _error_detect "apt-get -y install gcc" _error_detect "apt-get -y install make" _error_detect "apt-get -y install libmnl-dev" _error_detect "apt-get -y install libelf-dev" ;; fedora) _error_detect "dnf -y install qrencode" _error_detect "dnf -y install bc" _error_detect "dnf -y install gcc" _error_detect "dnf -y install make" _error_detect "dnf -y install libmnl-devel" _error_detect "dnf -y install elfutils-libelf-devel" ;; centos) _error_detect "yum -y install epel-release" _error_detect "yum -y install qrencode" _error_detect "yum -y install bc" _error_detect "yum -y install gcc" _error_detect "yum -y install make" _error_detect "yum -y install yum-utils" [ -n "$(_os_ver)" -a "$(_os_ver)" -eq 8 ] && _error_detect "yum-config-manager --enable PowerTools" _error_detect "yum -y install libmnl-devel" _error_detect "yum -y install elfutils-libelf-devel" ;; *) ;; # do nothing esac } # Install from repository install_wg_1() { install_wg_pkgs _info "Install wireguard from repository" case "$(_os)" in ubuntu) _error_detect "add-apt-repository ppa:wireguard/wireguard" _error_detect "apt-get update" _error_detect "apt-get -y install linux-headers-$(uname -r)" _error_detect "apt-get -y install wireguard-dkms" _error_detect "apt-get -y install wireguard-tools" ;; debian) echo "deb http://deb.debian.org/debian/ unstable main" > /etc/apt/sources.list.d/unstable.list printf 'Package: *\nPin: release a=unstable\nPin-Priority: 90\n' > /etc/apt/preferences.d/limit-unstable _error_detect "apt-get update" _error_detect "apt-get -y install linux-headers-$(uname -r)" _error_detect "apt-get -y install wireguard-dkms" _error_detect "apt-get -y install wireguard-tools" ;; fedora) _error_detect "dnf -y copr enable jdoss/wireguard" _error_detect "dnf -y install kernel-devel" _error_detect "dnf -y install kernel-headers" _error_detect "dnf -y install wireguard-dkms" _error_detect "dnf -y install wireguard-tools" ;; centos) if [ -n "$(_os_ver)" -a "$(_os_ver)" -eq 7 ]; then _error_detect "curl -Lso /etc/yum.repos.d/wireguard.repo https://copr.fedorainfracloud.org/coprs/jdoss/wireguard/repo/epel-7/jdoss-wireguard-epel-7.repo" fi if [ -n "$(_os_ver)" -a "$(_os_ver)" -eq 8 ]; then _error_detect "curl -Lso /etc/yum.repos.d/wireguard.repo https://copr.fedorainfracloud.org/coprs/jdoss/wireguard/repo/epel-8/jdoss-wireguard-epel-8.repo" fi _error_detect "yum -y install kernel-devel" _error_detect "yum -y install kernel-headers" _error_detect "yum -y install wireguard-dkms" _error_detect "yum -y install wireguard-tools" ;; *) ;; # do nothing esac } # Install from source install_wg_2() { install_wg_pkgs _info "Install wireguard from source" case "$(_os)" in ubuntu|debian|raspbian) if [ ! -d "/usr/src/linux-headers-$(uname -r)" ]; then if [ "$(_os)" = "raspbian" ]; then _error_detect "apt-get -y install raspberrypi-kernel-headers" else _error_detect "apt-get -y install linux-headers-$(uname -r)" fi fi ;; fedora) [ ! -d "/usr/src/kernels/$(uname -r)" ] && _error_detect "dnf -y install kernel-headers" && _error_detect "dnf -y install kernel-devel" ;; centos) [ ! -d "/usr/src/kernels/$(uname -r)" ] && _error_detect "yum -y install kernel-headers" && _error_detect "yum -y install kernel-devel" ;; *) ;; # do nothing esac install_wg_module install_wg_tools } # Install wireguard tools from repo install_wg_3() { install_wg_pkgs _info "Install wireguard from repository" case "$(_os)" in ubuntu) _error_detect "add-apt-repository ppa:wireguard/wireguard" _error_detect "apt-get update" _error_detect "apt-get -y install --no-install-recommends wireguard-tools" ;; debian) echo "deb http://deb.debian.org/debian/ unstable main" > /etc/apt/sources.list.d/unstable.list printf 'Package: *\nPin: release a=unstable\nPin-Priority: 90\n' > /etc/apt/preferences.d/limit-unstable _error_detect "apt-get update" _error_detect "apt-get -y install --no-install-recommends wireguard-tools" ;; fedora) _error_detect "dnf -y copr enable jdoss/wireguard" _error_detect "dnf -y install wireguard-tools" ;; centos) if [ -n "$(_os_ver)" -a "$(_os_ver)" -eq 7 ]; then _error_detect "curl -Lso /etc/yum.repos.d/wireguard.repo https://copr.fedorainfracloud.org/coprs/jdoss/wireguard/repo/epel-7/jdoss-wireguard-epel-7.repo" fi if [ -n "$(_os_ver)" -a "$(_os_ver)" -eq 8 ]; then _error_detect "curl -Lso /etc/yum.repos.d/wireguard.repo https://copr.fedorainfracloud.org/coprs/jdoss/wireguard/repo/epel-8/jdoss-wireguard-epel-8.repo" fi _error_detect "yum -y install wireguard-tools" ;; *) ;; # do nothing esac } # Install wireguard tools from source install_wg_4() { install_wg_pkgs _info "Install wireguard tools from source" install_wg_tools } # Uninstall WireGuard uninstall_wg() { if ! _is_installed; then _error "WireGuard is not installed" fi _info "Uninstall WireGuard start" # stop wireguard at first _error_detect "systemctl stop wg-quick@${SERVER_WG_NIC}" _error_detect "systemctl disable wg-quick@${SERVER_WG_NIC}" # if wireguard has been installed from repository if _exists "yum" && _exists "rpm"; then if rpm -qa | grep -q wireguard-dkms; then _error_detect "yum -y remove wireguard-dkms" fi if rpm -qa | grep -q wireguard-tools; then _error_detect "yum -y remove wireguard-tools" fi elif _exists "apt" && _exists "apt-get"; then if apt list --installed | grep -q wireguard-dkms; then _error_detect "apt-get -y remove wireguard-dkms" fi if apt list --installed | grep -q wireguard-tools; then _error_detect "apt-get -y remove wireguard-tools" fi fi # if wireguard has been installed from source if _is_installed; then _error_detect "rm -f /usr/bin/wg" _error_detect "rm -f /usr/bin/wg-quick" _error_detect "rm -f /usr/share/man/man8/wg.8" _error_detect "rm -f /usr/share/man/man8/wg-quick.8" _exists "modprobe" && _error_detect "modprobe -r wireguard" fi [ -d "/etc/wireguard" ] && _error_detect "rm -fr /etc/wireguard" _info "Uninstall WireGuard completed" } # Create server interface create_server_if() { SERVER_PRIVATE_KEY="$(wg genkey)" SERVER_PUBLIC_KEY="$(echo ${SERVER_PRIVATE_KEY} | wg pubkey)" CLIENT_PRIVATE_KEY="$(wg genkey)" CLIENT_PUBLIC_KEY="$(echo ${CLIENT_PRIVATE_KEY} | wg pubkey)" CLIENT_PRE_SHARED_KEY="$( wg genpsk )" _info "Create server interface: /etc/wireguard/${SERVER_WG_NIC}.conf" [ ! -d "/etc/wireguard" ] && mkdir -p "/etc/wireguard" if [ -n "${SERVER_PUB_IPV6}" ]; then cat > /etc/wireguard/${SERVER_WG_NIC}.conf < /etc/wireguard/${SERVER_WG_NIC}.conf < /etc/wireguard/${SERVER_WG_NIC}_client < /etc/wireguard/${SERVER_WG_NIC}_client <> /etc/sysctl.conf [ -n "${SERVER_PUB_IPV6}" ] && echo "net.ipv6.conf.all.forwarding = 1" >> /etc/sysctl.conf sysctl -p >/dev/null 2>&1 } # Set firewall rules set_firewall() { _info "Setting firewall rules" if _exists "firewall-cmd"; then if firewall-cmd --state > /dev/null 2>&1; then default_zone="$(firewall-cmd --get-default-zone)" if [ "$(firewall-cmd --zone=${default_zone} --query-masquerade)" = "no" ]; then _error_detect "firewall-cmd --permanent --zone=${default_zone} --add-masquerade" fi if ! firewall-cmd --list-ports | grep -qw "${SERVER_WG_PORT}/udp"; then _error_detect "firewall-cmd --permanent --zone=${default_zone} --add-port=${SERVER_WG_PORT}/udp" fi _error_detect "firewall-cmd --reload" else _warn "Firewalld service unit is not running, please start it and manually set" _warn "Maybe you need to run these commands like below:" _warn "systemctl start firewalld" _warn "firewall-cmd --permanent --zone=public --add-masquerade" _warn "firewall-cmd --permanent --zone=public --add-port=${SERVER_WG_PORT}/udp" _warn "firewall-cmd --reload" fi else if _exists "iptables"; then iptables -A INPUT -p udp --dport ${SERVER_WG_PORT} -j ACCEPT iptables -A FORWARD -i ${SERVER_WG_NIC} -j ACCEPT iptables -t nat -A POSTROUTING -o ${SERVER_PUB_NIC} -j MASQUERADE iptables-save > /etc/iptables.rules if [ -d "/etc/network/if-up.d" ]; then cat > /etc/network/if-up.d/iptables < /etc/ip6tables.rules if [ -d "/etc/network/if-up.d" ]; then cat > /etc/network/if-up.d/ip6tables < ${new_client_if} <> ${default_server_if} < ${new_client_if} <> ${default_server_if} < /dev/null 2>&1; then restart_flg=0 get_latest_module_ver wg_ver="$(echo ${wireguard_ver} | grep -oE '[0-9.]+')" _info "wireguard-dkms version: $(_green ${installed_wg_ver})" _info "wireguard-dkms latest version: $(_green ${wg_ver})" if check_kernel_version; then _info "wireguard-dkms has been merged into Linux >= 5.6 and therefore this compatibility module is no longer required" else if _version_gt "${wg_ver}" "${installed_wg_ver}"; then _info "Starting upgrade wireguard-dkms" install_wg_module _info "Update wireguard-dkms completed" restart_flg=1 else _info "There is no update available for wireguard-dkms" fi fi get_latest_tools_ver wg_tools_ver="$(echo ${wireguard_tools_ver} | grep -oE '[0-9.]+')" _info "wireguard-tools version: $(_green ${installed_wg_tools_ver})" _info "wireguard-tools latest version: $(_green ${wg_tools_ver})" if _version_gt "${wg_tools_ver}" "${installed_wg_tools_ver}"; then _info "Starting upgrade wireguard-tools" install_wg_tools _info "Update wireguard-tools completed" restart_flg=1 else _info "There is no update available for wireguard-tools" fi if [ ${restart_flg} -eq 1 ]; then _error_detect "systemctl daemon-reload" _error_detect "systemctl restart wg-quick@${SERVER_WG_NIC}" fi else _red "WireGuard was not installed, maybe you need to install it at first\n" fi } main() { action="$1" [ -z "${action}" ] && show_help && exit 0 case "${action}" in -h|--help) show_help ;; -r|--repo) install_from_repo ;; -s|--source) install_from_source ;; -u|--update) update_from_source ;; -v|--version) check_version ;; -a|--add) add_client ;; -d|--del) remove_client ;; -l|--list) list_clients ;; -n|--uninstall) uninstall_wg ;; *) show_help ;; esac } SERVER_PUB_IPV4="${VPN_SERVER_PUB_IPV4:-$(_ipv4)}" SERVER_PUB_IPV6="${VPN_SERVER_PUB_IPV6:-$(_ipv6)}" SERVER_PUB_NIC="${VPN_SERVER_PUB_NIC:-$(_nic)}" SERVER_WG_NIC="${VPN_SERVER_WG_NIC:-wg0}" SERVER_WG_IPV4="${VPN_SERVER_WG_IPV4:-10.88.88.1}" SERVER_WG_IPV6="${VPN_SERVER_WG_IPV6:-fd88:88:88::1}" SERVER_WG_PORT="${VPN_SERVER_WG_PORT:-$(_port)}" CLIENT_WG_IPV4="${VPN_CLIENT_WG_IPV4:-10.88.88.2}" CLIENT_WG_IPV6="${VPN_CLIENT_WG_IPV6:-fd88:88:88::2}" CLIENT_DNS_1="${VPN_CLIENT_DNS_1:-1.1.1.1}" CLIENT_DNS_2="${VPN_CLIENT_DNS_2:-8.8.8.8}" main "$@"