mirror of
https://github.com/oneclickvirt/pve.git
synced 2025-09-06 14:44:52 +08:00
349 lines
14 KiB
Bash
349 lines
14 KiB
Bash
#!/bin/bash
|
||
# from
|
||
# https://github.com/oneclickvirt/pve
|
||
# 2024.03.12
|
||
# 自动选择要绑定的IPV6地址
|
||
# ./buildvm_onlyv6.sh VMID 用户名 密码 CPU核数 内存 硬盘 系统 存储盘
|
||
# ./buildvm_onlyv6.sh 152 test1 1234567 1 512 5 debian11 local
|
||
|
||
cd /root >/dev/null 2>&1
|
||
# 创建NAT的虚拟机
|
||
vm_num="${1:-102}"
|
||
user="${2:-test}"
|
||
password="${3:-123456}"
|
||
core="${4:-1}"
|
||
memory="${5:-512}"
|
||
disk="${6:-5}"
|
||
system="${7:-ubuntu22}"
|
||
storage="${8:-local}"
|
||
rm -rf "vm$name"
|
||
|
||
_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
|
||
_yellow "No UTF-8 locale found"
|
||
else
|
||
export LC_ALL="$utf8_locale"
|
||
export LANG="$utf8_locale"
|
||
export LANGUAGE="$utf8_locale"
|
||
_green "Locale set to $utf8_locale"
|
||
fi
|
||
if [ ! -f /usr/local/bin/pve_check_ipv6 ]; then
|
||
_yellow "No ipv6 address exists to open a server with a standalone IPV6 address"
|
||
fi
|
||
if ! grep -q "vmbr2" /etc/network/interfaces; then
|
||
_yellow "No vmbr2 exists to open a server with a standalone IPV6 address"
|
||
fi
|
||
|
||
# 检测ndppd服务是否启动了
|
||
service_status=$(systemctl is-active ndpresponder.service)
|
||
if [ "$service_status" == "active" ]; then
|
||
_green "The ndpresponder service started successfully and is running, and the host can open a service with a separate IPV6 address."
|
||
_green "ndpresponder服务启动成功且正在运行,宿主机可开设带独立IPV6地址的服务。"
|
||
else
|
||
_green "The status of the ndpresponder service is abnormal and the host may not open a service with a separate IPV6 address."
|
||
_green "ndpresponder服务状态异常,宿主机不可开设带独立IPV6地址的服务。"
|
||
exit 1
|
||
fi
|
||
|
||
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
|
||
}
|
||
|
||
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://cdn0.spiritlhl.top/" "http://cdn3.spiritlhl.net/" "http://cdn1.spiritlhl.net/" "https://ghproxy.com/" "http://cdn2.spiritlhl.net/")
|
||
if [ ! -d "qcow" ]; then
|
||
mkdir qcow
|
||
fi
|
||
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
|
||
fi
|
||
if [ "$system_arch" = "x86" ]; then
|
||
file_path=""
|
||
# 过去手动修补的镜像
|
||
old_images=(
|
||
"debian10"
|
||
"debian11"
|
||
"debian12"
|
||
"ubuntu18"
|
||
"ubuntu20"
|
||
"ubuntu22"
|
||
"centos7"
|
||
"archlinux"
|
||
"almalinux8"
|
||
"fedora33"
|
||
"fedora34"
|
||
"opensuse-leap-15"
|
||
"alpinelinux_edge"
|
||
"alpinelinux_stable"
|
||
"rockylinux8"
|
||
"centos8-stream"
|
||
)
|
||
# 新的自动修补的镜像
|
||
# response=$(curl -sSL -m 6 -H "Accept: application/vnd.github.v3+json" "https://api.github.com/repos/oneclickvirt/pve_kvm_images/releases/tags/images")
|
||
# # 如果 https://api.github.com/ 请求失败,则使用 https://githubapi.spiritlhl.workers.dev/ ,此时可能宿主机无IPV4网络
|
||
# if [ -z "$response" ]; then
|
||
# response=$(curl -sSL -m 6 -H "Accept: application/vnd.github.v3+json" "https://githubapi.spiritlhl.workers.dev/repos/oneclickvirt/pve_kvm_images/releases/tags/images")
|
||
# fi
|
||
# # 如果 https://githubapi.spiritlhl.workers.dev/ 请求失败,则使用 https://githubapi.spiritlhl.top/ ,此时可能宿主机在国内
|
||
# if [ -z "$response" ]; then
|
||
# response=$(curl -sSL -m 6 -H "Accept: application/vnd.github.v3+json" "https://githubapi.spiritlhl.top/repos/oneclickvirt/pve_kvm_images/releases/tags/images")
|
||
# fi
|
||
# if [[ -n "$response" ]]; then
|
||
# new_images=($(echo "$response" | grep -oP '"name": "\K[^"]+' | grep 'qcow2' | awk '{print $1}'))
|
||
# for ((i=0; i<${#new_images[@]}; i++)); do
|
||
# new_images[i]=${new_images[i]%.qcow2}
|
||
# done
|
||
# combined=($(echo "${old_images[@]}" "${new_images[@]}" | tr ' ' '\n' | sort -u))
|
||
# systems=("${combined[@]}")
|
||
# else
|
||
# systems=("${old_images[@]}")
|
||
# fi
|
||
new_images=($(curl -slk -m 6 https://down.idc.wiki/Image/realServer-Template/current/qcow2/ | grep -o '<a href="[^"]*">' | awk -F'"' '{print $2}' | sed -n '/qcow2$/s#/Image/realServer-Template/current/qcow2/##p'))
|
||
if [[ -n "$new_images" ]]; then
|
||
for ((i=0; i<${#new_images[@]}; i++)); do
|
||
new_images[i]=${new_images[i]%.qcow2}
|
||
done
|
||
combined=($(echo "${old_images[@]}" "${new_images[@]}" | tr ' ' '\n' | sort -u))
|
||
systems=("${combined[@]}")
|
||
else
|
||
systems=("${old_images[@]}")
|
||
fi
|
||
for sys in ${systems[@]}; do
|
||
if [[ "$system" == "$sys" ]]; then
|
||
file_path="/root/qcow/${system}.qcow2"
|
||
break
|
||
fi
|
||
done
|
||
if [[ -z "$file_path" ]]; then
|
||
_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
|
||
if [ ! -f "$file_path" ]; then
|
||
check_cdn_file
|
||
ver=""
|
||
# 使用新镜像,自动修补版本
|
||
if [[ -n "$new_images" ]]; then
|
||
for image in "${new_images[@]}"; do
|
||
if [[ " ${image} " == *" $system "* ]]; then
|
||
ver="auto_build"
|
||
url="${cdn_success_url}https://github.com/oneclickvirt/pve_kvm_images/releases/download/images/${image}.qcow2"
|
||
curl -Lk -o "$file_path" "$url"
|
||
if [ $? -ne 0 ]; then
|
||
_red "Failed to download $file_path"
|
||
ver=""
|
||
rm -rf "$file_path"
|
||
break
|
||
else
|
||
_blue "Use auto-fixed image: ${image}"
|
||
break
|
||
fi
|
||
fi
|
||
done
|
||
fi
|
||
# 使用旧镜像,手动修补版本
|
||
if [[ -z "$ver" ]]; then
|
||
v20=("fedora34" "almalinux8" "debian11" "debian12" "ubuntu18" "ubuntu20" "ubuntu22" "centos7" "alpinelinux_edge" "alpinelinux_stable" "rockylinux8")
|
||
v11=("ubuntu18" "ubuntu20" "ubuntu22" "debian10" "debian11")
|
||
v10=("almalinux8" "archlinux" "fedora33" "opensuse-leap-15" "ubuntu18" "ubuntu20" "ubuntu22" "debian10" "debian11")
|
||
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
|
||
if [[ "$system" == "centos8-stream" ]]; then
|
||
url="https://api.ilolicon.com/centos8-stream.qcow2"
|
||
curl -Lk -o "$file_path" "$url"
|
||
if [ $? -ne 0 ]; then
|
||
_red "Unable to download corresponding system, please check https://github.com/oneclickvirt/kvm_images/ for supported system images "
|
||
_red "无法下载对应系统,请查看 https://github.com/oneclickvirt/kvm_images/ 支持的系统镜像 "
|
||
rm -rf "$file_path"
|
||
exit 1
|
||
else
|
||
_blue "Use manual-fixed image: ${system}"
|
||
break
|
||
fi
|
||
else
|
||
if [[ -n "$ver" ]]; then
|
||
url="${cdn_success_url}https://github.com/oneclickvirt/kvm_images/releases/download/${ver}/${system}.qcow2"
|
||
curl -Lk -o "$file_path" "$url"
|
||
if [ $? -ne 0 ]; then
|
||
_red "Unable to download corresponding system, please check https://github.com/oneclickvirt/kvm_images/ for supported system images "
|
||
_red "无法下载对应系统,请查看 https://github.com/oneclickvirt/kvm_images/ 支持的系统镜像 "
|
||
rm -rf "$file_path"
|
||
exit 1
|
||
else
|
||
_blue "Use manual-fixed image: ${system}"
|
||
break
|
||
fi
|
||
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
|
||
fi
|
||
fi
|
||
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
|
||
fi
|
||
first_digit=${vm_num:0:1}
|
||
second_digit=${vm_num:1:1}
|
||
third_digit=${vm_num:2:1}
|
||
if [ $first_digit -le 2 ]; then
|
||
if [ $second_digit -eq 0 ]; then
|
||
num=$third_digit
|
||
else
|
||
num=$second_digit$third_digit
|
||
fi
|
||
else
|
||
num=$((first_digit - 2))$second_digit$third_digit
|
||
fi
|
||
# 检测IPV6相关的信息
|
||
if [ -f /usr/local/bin/pve_check_ipv6 ]; then
|
||
host_ipv6_address=$(cat /usr/local/bin/pve_check_ipv6)
|
||
ipv6_address_without_last_segment="${host_ipv6_address%:*}:"
|
||
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
|
||
qm create $vm_num --agent 1 --scsihw virtio-scsi-single --serial0 socket --cores $core --sockets 1 --cpu host --net0 virtio,bridge=vmbr1,firewall=0 --net1 virtio,bridge=vmbr2,firewall=0
|
||
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}
|
||
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
|
||
qm set $vm_num --scsihw virtio-scsi-pci --scsi0 ${storage}:${vm_num}/vm-${vm_num}-disk-0.raw
|
||
fi
|
||
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
|
||
qm set $vm_num --nameserver 1.1.1.1
|
||
# qm set $vm_num --nameserver 1.0.0.1
|
||
qm set $vm_num --searchdomain local
|
||
user_ip="172.16.1.${num}"
|
||
qm set $vm_num --ipconfig0 ip=${user_ip}/24,gw=172.16.1.1
|
||
qm set $vm_num --ipconfig1 ip6="${ipv6_address_without_last_segment}${vm_num}/128",gw6="${host_ipv6_address}"
|
||
qm set $vm_num --cipassword $password --ciuser $user
|
||
sleep 5
|
||
qm resize $vm_num scsi0 ${disk}G
|
||
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
|
||
qm start $vm_num
|
||
echo "$vm_num $user $password $core $memory $disk $system $storage ${ipv6_address_without_last_segment}${vm_num}" >>"vm${vm_num}"
|
||
# 虚拟机的相关信息将会存储到对应的虚拟机的NOTE中,可在WEB端查看
|
||
data=$(echo " VMID 用户名-username 密码-password CPU核数-CPU 内存-memory 硬盘-disk 系统-system 存储盘-storage 外网IPV6-ipv6")
|
||
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
|
||
echo "${data_array[$i]} ${values_array[$i]}"
|
||
echo ""
|
||
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}"
|