pve/scripts/buildvm_manual_ip.sh

338 lines
12 KiB
Bash
Raw Normal View History

2023-06-20 12:21:09 +08:00
#!/bin/bash
# from
# https://github.com/spiritLHLS/pve
2023-08-04 09:47:53 +08:00
# 2023.08.04
2023-06-20 12:21:09 +08:00
# 手动指定要绑定的IPV4地址
2023-06-29 21:41:08 +08:00
2023-06-20 12:21:09 +08:00
# ./buildvm_manual_ip.sh VMID 用户名 密码 CPU核数 内存 硬盘 系统 存储盘 IPV4地址
2023-06-20 12:29:37 +08:00
# ./buildvm_manual_ip.sh 152 test1 1234567 1 512 5 debian11 local a.b.c.d/24
2023-06-20 12:21:09 +08:00
cd /root >/dev/null 2>&1
# 创建独立IPV4地址的虚拟机
vm_num="${1:-152}"
user="${2:-test}"
password="${3:-123456}"
core="${4:-1}"
memory="${5:-512}"
disk="${6:-5}"
2023-07-31 11:02:23 +08:00
system="${7:-ubuntu22}"
2023-06-20 12:21:09 +08:00
storage="${8:-local}"
extra_ip="${9}"
rm -rf "vm$name"
user_ip=""
user_ip_range=""
gateway=""
2023-06-20 12:41:24 +08:00
_red() { echo -e "\033[31m\033[01m$@\033[0m"; }
_green() { echo -e "\033[32m\033[01m$@\033[0m"; }
_yellow() { echo -e "\033[33m\033[01m$@\033[0m"; }
_blue() { echo -e "\033[36m\033[01m$@\033[0m"; }
reading(){ read -rp "$(_green "$1")" "$2"; }
utf8_locale=$(locale -a 2>/dev/null | grep -i -m 1 -E "utf8|UTF-8")
if [[ -z "$utf8_locale" ]]; then
2023-06-29 21:41:08 +08:00
_yellow "No UTF-8 locale found"
2023-06-20 12:41:24 +08:00
else
2023-06-29 21:41:08 +08:00
export LC_ALL="$utf8_locale"
export LANG="$utf8_locale"
export LANGUAGE="$utf8_locale"
_green "Locale set to $utf8_locale"
2023-06-20 12:41:24 +08:00
fi
2023-07-31 10:17:41 +08:00
get_system_arch() {
local sysarch="$(uname -m)"
if [ "${sysarch}" = "unknown" ] || [ "${sysarch}" = "" ]; then
local sysarch="$(arch)"
fi
# 根据架构信息设置系统位数并下载文件,其余 * 包括了 x86_64
case "${sysarch}" in
"i386" | "i686" | "x86_64")
system_arch="x86"
;;
"armv7l" | "armv8" | "armv8l" | "aarch64")
system_arch="arch"
;;
*)
system_arch=""
;;
esac
}
2023-06-20 12:21:09 +08:00
is_ipv4() {
local ip=$1
local regex="^([0-9]{1,3}\.){3}[0-9]{1,3}$"
if [[ $ip =~ $regex ]]; then
return 0 # 符合IPv4格式
else
return 1 # 不符合IPv4格式
fi
}
if [[ -z "$extra_ip" ]]; then
2023-06-29 21:41:08 +08:00
_yellow "No IPV4 address is manually assigned"
_yellow "IPV4地址未手动指定"
exit 1
2023-06-20 12:21:09 +08:00
else
2023-06-29 21:41:08 +08:00
user_ip=$(echo "$extra_ip" | cut -d'/' -f1)
user_ip_range=$(echo "$extra_ip" | cut -d'/' -f2)
if is_ipv4 "$user_ip"; then
_green "This IPV4 address will be used: ${user_ip}"
_green "将使用此IPV4地址: ${user_ip}"
else
_yellow "IPV4 addresses do not conform to the rules"
_yellow "IPV4地址不符合规则"
exit 1
fi
2023-06-20 12:21:09 +08:00
fi
check_cdn() {
local o_url=$1
for cdn_url in "${cdn_urls[@]}"; do
if curl -sL -k "$cdn_url$o_url" --max-time 6 | grep -q "success" > /dev/null 2>&1; then
export cdn_success_url="$cdn_url"
return
fi
sleep 0.5
done
export cdn_success_url=""
}
check_cdn_file() {
check_cdn "https://raw.githubusercontent.com/spiritLHLS/ecs/main/back/test"
if [ -n "$cdn_success_url" ]; then
_yellow "CDN available, using CDN"
else
_yellow "No CDN available, no use CDN"
fi
}
cdn_urls=("https://cdn.spiritlhl.workers.dev/" "https://cdn3.spiritlhl.net/" "https://cdn1.spiritlhl.net/" "https://ghproxy.com/" "https://cdn2.spiritlhl.net/")
if [ ! -d "qcow" ]; then
mkdir qcow
fi
2023-07-31 10:17:41 +08:00
get_system_arch
if [ -z "${system_arch}" ] || [ ! -v system_arch ]; then
_red "This script can only run on machines under x86_64 or arm architecture."
exit 1
2023-06-20 12:21:09 +08:00
fi
2023-07-31 10:17:41 +08:00
if [ "$system_arch" = "x86" ]; then
2023-08-02 10:07:18 +08:00
file_path=""
systems=(
"debian10"
"debian11"
"debian12"
"ubuntu18"
"ubuntu20"
"ubuntu22"
"centos7"
2023-08-02 11:50:21 +08:00
"archlinux"
"almalinux8"
2023-08-02 12:08:33 +08:00
"fedora33"
2023-08-02 12:11:07 +08:00
"fedora34"
2023-08-02 10:07:18 +08:00
"opensuse-leap-15"
2023-08-02 19:03:56 +08:00
"alpinelinux_edge"
"alpinelinux_stable"
2023-08-02 19:55:19 +08:00
"rockylinux8"
2023-08-03 16:34:12 +08:00
"centos8-stream"
2023-08-02 10:07:18 +08:00
)
2023-07-31 10:17:41 +08:00
for sys in ${systems[@]}; do
if [[ "$system" == "$sys" ]]; then
file_path="/root/qcow/${system}.qcow2"
break
fi
done
if [[ -z "$file_path" ]]; then
2023-08-01 15:46:12 +08:00
_red "Unable to install corresponding system, please check https://github.com/oneclickvirt/kvm_images/ for supported system images "
_red "无法安装对应系统,请查看 https://github.com/oneclickvirt/kvm_images/ 支持的系统镜像 "
2023-07-31 10:17:41 +08:00
exit 1
fi
if [ ! -f "$file_path" ]; then
check_cdn_file
2023-08-02 10:07:18 +08:00
ver=""
2023-08-02 19:55:19 +08:00
v20=("fedora34" "almalinux8" "debian11" "debian12" "ubuntu18" "ubuntu20" "ubuntu22" "centos7" "alpinelinux_edge" "alpinelinux_stable" "rockylinux8")
2023-08-02 11:55:42 +08:00
v11=("ubuntu18" "ubuntu20" "ubuntu22" "debian10" "debian11")
2023-08-02 12:46:21 +08:00
v10=("almalinux8" "archlinux" "fedora33" "opensuse-leap-15" "ubuntu18" "ubuntu20" "ubuntu22" "debian10" "debian11")
2023-08-02 10:07:18 +08:00
ver_list=(v20 v11 v10)
ver_name_list=("v2.0" "v1.1" "v1.0")
for ver in "${ver_list[@]}"; do
array_name="${ver}[@]"
array=("${!array_name}")
if [[ " ${array[*]} " == *" $system "* ]]; then
index=$(echo ${ver_list[*]} | tr -s ' ' '\n' | grep -n "$ver" | cut -d':' -f1)
ver="${ver_name_list[$((index-1))]}"
break
fi
done
2023-08-03 16:56:28 +08:00
if [[ "$system" == "centos8-stream" ]]; then
2023-08-03 16:34:12 +08:00
url="https://git.ilolicon.dev/Ella-Alinda/images/raw/branch/main/centos8-stream.qcow2"
2023-08-03 16:40:44 +08:00
curl -Lk -o "$file_path" "$url"
2023-08-02 10:07:18 +08:00
else
2023-08-03 16:34:12 +08:00
if [[ -n "$ver" ]]; then
url="${cdn_success_url}https://github.com/oneclickvirt/kvm_images/releases/download/${ver}/${system}.qcow2"
2023-08-03 16:40:44 +08:00
curl -Lk -o "$file_path" "$url"
2023-08-03 16:34:12 +08:00
else
_red "Unable to install corresponding system, please check https://github.com/oneclickvirt/kvm_images/ for supported system images "
_red "无法安装对应系统,请查看 https://github.com/oneclickvirt/kvm_images/ 支持的系统镜像 "
exit 1
fi
2023-08-02 10:07:18 +08:00
fi
2023-07-31 10:17:41 +08:00
fi
elif [ "$system_arch" = "arch" ]; then
systems=("ubuntu14" "ubuntu16" "ubuntu18" "ubuntu20" "ubuntu22")
for sys in ${systems[@]}; do
if [[ "$system" == "$sys" ]]; then
file_path="/root/qcow/${system}.img"
break
fi
done
if [[ -z "$file_path" ]]; then
# https://www.debian.org/mirror/list
_red "Unable to install corresponding system, please check http://cloud-images.ubuntu.com for supported system images "
_red "无法安装对应系统,请查看 http://cloud-images.ubuntu.com 支持的系统镜像 "
exit 1
fi
if [ -n "$file_path" ] && [ -f "$file_path" ]; then
case "$system" in
ubuntu14)
version="trusty"
;;
ubuntu16)
version="xenial"
;;
ubuntu18)
version="bionic"
;;
ubuntu20)
version="focal"
;;
ubuntu22)
version="jammy"
;;
*)
echo "Unsupported Ubuntu version."
exit 1
;;
esac
url="http://cloud-images.ubuntu.com/${version}/current/${version}-server-cloudimg-arm64.img"
curl -L -o "$file_path" "$url"
fi
2023-06-20 12:21:09 +08:00
fi
# 查询信息
if ! command -v lshw > /dev/null 2>&1; then
2023-06-29 21:41:08 +08:00
apt-get install -y lshw
2023-06-20 12:21:09 +08:00
fi
if ! command -v ping > /dev/null 2>&1; then
2023-06-29 21:41:08 +08:00
apt-get install -y iputils-ping
apt-get install -y ping
2023-06-20 12:21:09 +08:00
fi
interface=$(lshw -C network | awk '/logical name:/{print $3}' | head -1)
user_main_ip_range=$(grep -A 1 "iface ${interface}" /etc/network/interfaces | grep "address" | awk '{print $2}' | head -n 1)
if [ -z "$user_main_ip_range" ]; then
2023-06-29 21:41:08 +08:00
_red "Host available IP interval query failed"
_red "宿主机可用IP区间查询失败"
exit 1
2023-06-20 12:21:09 +08:00
fi
# 宿主机的网关
gateway=$(grep -E "iface $interface" -A 3 "/etc/network/interfaces" | grep "gateway" | awk '{print $2}' | head -n 1)
if [ -z "$gateway" ]; then
2023-06-29 21:41:08 +08:00
_red "Host gateway query failed"
_red "宿主机网关查询失败"
exit 1
2023-06-20 12:21:09 +08:00
fi
# echo "ip=${user_ip}/${user_ip_range},gw=${gateway}"
# 检查变量是否为空并执行相应操作
if [ -z "$user_ip" ]; then
2023-06-29 21:41:08 +08:00
_red "Available IP match failed"
_red "可使用的IP匹配失败"
exit 1
2023-06-20 12:21:09 +08:00
fi
if [ -z "$user_ip_range" ]; then
2023-06-29 21:41:08 +08:00
_red "Available subnet size match failed"
_red "可使用的子网大小匹配失败"
exit 1
2023-06-20 12:21:09 +08:00
fi
2023-06-29 21:41:08 +08:00
_green "The current IP to which the VM will be bound is: ${user_ip}"
2023-06-20 12:21:09 +08:00
_green "当前虚拟机将绑定的IP为${user_ip}"
2023-08-04 09:47:53 +08:00
# 检测IPV6相关的信息
if [ -f /usr/local/bin/pve_check_ipv6 ]; then
ipv6_address=$(cat /usr/local/bin/pve_check_ipv6)
IFS="/" read -ra parts <<< "$ipv6_address"
part_1="${parts[0]}"
part_2="${parts[1]}"
IFS=":" read -ra part_1_parts <<< "$part_1"
part_1_last="${part_1_parts[-1]}"
if [ "$part_1_last" = "$vm_num" ]; then
ipv6_address=""
else
part_1_head=$(echo "$part_1" | awk -F':' 'BEGIN {OFS=":"} {last=""; for (i=1; i<NF; i++) {last=last $i ":"}; print last}')
ipv6_address="${part_1_head}${vm_num}"
fi
fi
if [ -f /usr/local/bin/pve_ipv6_prefixlen ]; then
ipv6_prefixlen=$(cat /usr/local/bin/pve_ipv6_prefixlen)
fi
if [ -f /usr/local/bin/pve_ipv6_gateway ]; then
ipv6_gateway=$(cat /usr/local/bin/pve_ipv6_gateway)
fi
2023-06-20 12:21:09 +08:00
qm create $vm_num --agent 1 --scsihw virtio-scsi-single --serial0 socket --cores $core --sockets 1 --cpu host --net0 virtio,bridge=vmbr0,firewall=0
2023-07-31 10:17:41 +08:00
if [ "$system_arch" = "x86" ]; then
qm importdisk $vm_num /root/qcow/${system}.qcow2 ${storage}
else
qm set $vm_num --bios ovmf
qm importdisk $vm_num /root/qcow/${system}.img ${storage}
2023-08-03 16:56:28 +08:00
fi
sleep 3
raw_name=$(ls /var/lib/vz/images/${vm_num}/*.raw | xargs -n1 basename | tail -n 1)
if [ -n "$raw_name" ]; then
qm set $vm_num --scsihw virtio-scsi-pci --scsi0 ${storage}:${vm_num}/${raw_name}
else
2023-07-31 10:17:41 +08:00
qm set $vm_num --scsihw virtio-scsi-pci --scsi0 ${storage}:${vm_num}/vm-${vm_num}-disk-0.raw
fi
2023-06-20 12:21:09 +08:00
qm set $vm_num --bootdisk scsi0
qm set $vm_num --boot order=scsi0
qm set $vm_num --memory $memory
# --swap 256
qm set $vm_num --ide2 ${storage}:cloudinit
2023-08-04 11:22:36 +08:00
qm set $vm_num --nameserver 8.8.8.8,2001:4860:4860::8888
qm set $vm_num --searchdomain 8.8.4.4,2001:4860:4860::8844
2023-08-04 09:47:53 +08:00
if [ -z "$ipv6_address" ] || [ -z "$ipv6_prefixlen" ] || [ -z "$ipv6_gateway" ] || [ "$ipv6_prefixlen" -gt 112 ]; then
qm set $vm_num --ipconfig0 ip=${user_ip}/${user_ip_range},gw=${gateway}
else
2023-08-04 11:22:36 +08:00
qm set $vm_num --ipconfig0 ip=${user_ip}/${user_ip_range},gw=${gateway},ip6=${ipv6_address}/${ipv6_prefixlen},gw6=${ipv6_gateway}
2023-08-04 09:47:53 +08:00
fi
2023-06-20 12:21:09 +08:00
qm set $vm_num --cipassword $password --ciuser $user
2023-08-02 12:08:33 +08:00
sleep 5
2023-06-20 12:21:09 +08:00
qm resize $vm_num scsi0 ${disk}G
2023-08-02 11:14:52 +08:00
if [ $? -ne 0 ]; then
if [[ $disk =~ ^[0-9]+G$ ]]; then
dnum=${disk::-1}
disk_m=$((dnum * 1024))
qm resize $vm_num scsi0 ${disk_m}M
fi
fi
2023-06-20 12:21:09 +08:00
qm start $vm_num
# 虚拟机的相关信息将会存储到对应的虚拟机的NOTE中可在WEB端查看
2023-08-04 09:47:53 +08:00
if [ -z "$ipv6_address" ] || [ -z "$ipv6_prefixlen" ] || [ -z "$ipv6_gateway" ] || [ "$ipv6_prefixlen" -gt 112 ]; then
echo "$vm_num $user $password $core $memory $disk $system $storage $user_ip" >> "vm${vm_num}"
data=$(echo " VMID 用户名-username 密码-password CPU核数-CPU 内存-memory 硬盘-disk 系统-system 存储盘-storage 外网IP地址-ipv4")
else
echo "$vm_num $user $password $core $memory $disk $system $storage $user_ip $ipv6_address" >> "vm${vm_num}"
data=$(echo " VMID 用户名-username 密码-password CPU核数-CPU 内存-memory 硬盘-disk 系统-system 存储盘-storage 外网IPV4-ipv4 外网IPV6-ipv6")
fi
2023-06-20 12:21:09 +08:00
values=$(cat "vm${vm_num}")
IFS=' ' read -ra data_array <<< "$data"
IFS=' ' read -ra values_array <<< "$values"
length=${#data_array[@]}
for ((i=0; i<$length; i++))
do
2023-08-04 09:47:53 +08:00
echo "${data_array[$i]} ${values_array[$i]}"
echo ""
2023-06-20 12:21:09 +08:00
done > "/tmp/temp${vm_num}.txt"
sed -i 's/^/# /' "/tmp/temp${vm_num}.txt"
cat "/etc/pve/qemu-server/${vm_num}.conf" >> "/tmp/temp${vm_num}.txt"
cp "/tmp/temp${vm_num}.txt" "/etc/pve/qemu-server/${vm_num}.conf"
rm -rf "/tmp/temp${vm_num}.txt"
cat "vm${vm_num}"