#!/bin/bash #fonts color Green="\033[32m" Red="\033[31m" GreenBG="\033[42;37m" RedBG="\033[41;37m" Font="\033[0m" #notification information OK="${Green}[OK]${Font}" Error="${Red}[错误]${Font}" cur_path="$(pwd)" cur_arg=$@ COMPOSE="docker-compose" judge() { if [[ 0 -eq $? ]]; then echo -e "${OK} ${GreenBG} $1 完成 ${Font}" sleep 1 else echo -e "${Error} ${RedBG} $1 失败${Font}" exit 1 fi } rand() { local min=$1 local max=$(($2-$min+1)) local num=$(($RANDOM+1000000000)) echo $(($num%$max+$min)) } rand_string() { local lan=$1 if [[ `uname` == 'Linux' ]]; then echo "$(date +%s%N | md5sum | cut -c 1-${lan})" else echo "$(docker run -it --rm alpine sh -c "date +%s%N | md5sum | cut -c 1-${lan}")" fi } supervisorctl_restart() { local RES=`run_exec php "supervisorctl update $1"` if [ -z "$RES" ]; then run_exec php "supervisorctl restart $1" else echo -e "$RES" fi } check_docker() { docker --version &> /dev/null if [ $? -ne 0 ]; then echo -e "${Error} ${RedBG} 未安装 Docker!${Font}" exit 1 fi docker-compose version &> /dev/null if [ $? -ne 0 ]; then docker compose version &> /dev/null if [ $? -ne 0 ]; then echo -e "${Error} ${RedBG} 未安装 Docker-compose!${Font}" exit 1 fi COMPOSE="docker compose" fi if [[ -n `$COMPOSE version | grep -E "\sv*1"` ]]; then $COMPOSE version echo -e "${Error} ${RedBG} Docker-compose 版本过低,请升级至v2+!${Font}" exit 1 fi } check_node() { npm --version &> /dev/null if [ $? -ne 0 ]; then echo -e "${Error} ${RedBG} 未安装nodejs!${Font}" exit 1 fi } docker_name() { echo `$COMPOSE ps | awk '{print $1}' | grep "\-$1\-"` } run_compile() { local type=$1 check_node if [ ! -d "./node_modules" ]; then npm install fi run_exec php "php bin/run --mode=$type" supervisorctl_restart php # if [ "$type" = "prod" ]; then rm -rf "./public/js/build" npx mix --production else npx mix watch --hot fi } run_electron() { local argv=$@ check_node if [ ! -d "./node_modules" ]; then npm install fi if [ ! -d "./electron/node_modules" ]; then pushd electron npm install popd fi # if [ -d "./electron/dist" ]; then rm -rf "./electron/dist" fi if [ -d "./electron/public" ] && [ "$argv" != "--nobuild" ]; then rm -rf "./electron/public" fi mkdir -p ./electron/public cp ./electron/index.html ./electron/public/index.html # if [ "$argv" != "dev" ] && [ "$argv" != "--nobuild" ]; then npx mix --production -- --env --electron fi node ./electron/build.js $argv } run_exec() { local container=$1 local cmd=$2 local name=`docker_name $container` if [ -z "$name" ]; then echo -e "${Error} ${RedBG} 没有找到 $container 容器! ${Font}" exit 1 fi docker exec -it "$name" /bin/sh -c "$cmd" } run_mysql() { if [ "$1" = "backup" ]; then # 备份数据库 database=$(env_get DB_DATABASE) username=$(env_get DB_USERNAME) password=$(env_get DB_PASSWORD) mkdir -p ${cur_path}/docker/mysql/backup filename="${cur_path}/docker/mysql/backup/${database}_$(date "+%Y%m%d%H%M%S").sql.gz" run_exec mariadb "exec mysqldump --databases $database -u$username -p$password" | gzip > $filename judge "备份数据库" [ -f "$filename" ] && echo -e "备份文件:$filename" elif [ "$1" = "recovery" ]; then # 还原数据库 database=$(env_get DB_DATABASE) username=$(env_get DB_USERNAME) password=$(env_get DB_PASSWORD) mkdir -p ${cur_path}/docker/mysql/backup list=`ls -1 "${cur_path}/docker/mysql/backup" | grep ".sql.gz"` if [ -z "$list" ]; then echo -e "${Error} ${RedBG} 没有备份文件!${Font}" exit 1 fi echo "$list" read -rp "请输入备份文件名称还原:" inputname filename="${cur_path}/docker/mysql/backup/${inputname}" if [ ! -f "$filename" ]; then echo -e "${Error} ${RedBG} 备份文件:${inputname} 不存在! ${Font}" exit 1 fi container_name=`docker_name mariadb` if [ -z "$container_name" ]; then echo -e "${Error} ${RedBG} 没有找到 mariadb 容器! ${Font}" exit 1 fi docker cp $filename $container_name:/ run_exec mariadb "gunzip < /$inputname | mysql -u$username -p$password $database" run_exec php "php artisan migrate" judge "还原数据库" fi } env_get() { local key=$1 local value=`cat ${cur_path}/.env | grep "^$key=" | awk -F '=' '{print $2}'` echo "$value" } env_set() { local key=$1 local val=$2 local exist=`cat ${cur_path}/.env | grep "^$key="` if [ -z "$exist" ]; then echo "$key=$val" >> $cur_path/.env else if [[ `uname` == 'Linux' ]]; then sed -i "/^${key}=/c\\${key}=${val}" ${cur_path}/.env else docker run -it --rm -v ${cur_path}:/www alpine sh -c "sed -i "/^${key}=/c\\${key}=${val}" /www/.env" fi if [ $? -ne 0 ]; then echo -e "${Error} ${RedBG} 设置env参数失败!${Font}" exit 1 fi fi } env_init() { if [ ! -f ".env" ]; then cp .env.docker .env fi if [ -z "$(env_get DB_ROOT_PASSWORD)" ]; then env_set DB_ROOT_PASSWORD "$(rand_string 16)" fi if [ -z "$(env_get APP_ID)" ]; then env_set APP_ID "$(rand_string 6)" fi if [ -z "$(env_get APP_IPPR)" ]; then env_set APP_IPPR "10.$(rand 50 100).$(rand 100 200)" fi } arg_get() { local find="n" local value="" for var in $cur_arg; do if [[ "$find" == "y" ]]; then if [[ ! $var =~ "--" ]]; then value=$var fi break fi if [[ "--$1" == "$var" ]] || [[ "-$1" == "$var" ]]; then find="y" value="yes" fi done echo $value } is_arm() { local get_arch=`arch` if [[ $get_arch =~ "aarch" ]] || [[ $get_arch =~ "arm" ]]; then echo "yes" else echo "no" fi } #################################################################################### #################################################################################### #################################################################################### if [[ "$1" != "electron" ]]; then check_docker env_init fi if [ $# -gt 0 ]; then if [[ "$1" == "init" ]] || [[ "$1" == "install" ]]; then shift 1 # 判断架构 if [[ "$(is_arm)" == "yes" ]] && [[ -z "$(arg_get force)" ]]; then echo -e "${Error} ${RedBG}暂不支持arm架构,强制安装请使用:./cmd install --force${Font}" exit 1 fi # 初始化文件 if [[ -n "$(arg_get relock)" ]]; then rm -rf node_modules rm -rf package-lock.json rm -rf vendor rm -rf composer.lock fi mkdir -p "${cur_path}/docker/log/supervisor" mkdir -p "${cur_path}/docker/mysql/data" chmod -R 775 "${cur_path}/docker/log/supervisor" chmod -R 775 "${cur_path}/docker/mysql/data" # 启动容器 [[ "$(arg_get port)" -gt 0 ]] && env_set APP_PORT "$(arg_get port)" $COMPOSE up php -d # 安装composer依赖 run_exec php "composer install" if [ ! -f "${cur_path}/vendor/autoload.php" ]; then run_exec php "composer config repo.packagist composer https://packagist.phpcomposer.com" run_exec php "composer install" run_exec php "composer config --unset repos.packagist" fi if [ ! -f "${cur_path}/vendor/autoload.php" ]; then echo -e "${Error} ${RedBG}composer install 失败,请重试! ${Font}" exit 1 fi [[ -z "$(env_get APP_KEY)" ]] && run_exec php "php artisan key:generate" run_exec php "php bin/run --mode=prod" # 检查数据库 remaining=10 while [ ! -f "${cur_path}/docker/mysql/data/$(env_get DB_DATABASE)/db.opt" ]; do ((remaining=$remaining-1)) if [ $remaining -lt 0 ]; then echo -e "${Error} ${RedBG} 数据库安装失败! ${Font}" exit 1 fi chmod -R 775 "${cur_path}/docker/mysql/data" sleep 3 done run_exec php "php artisan migrate --seed" # 设置初始化密码 res=`run_exec mariadb "sh /etc/mysql/repassword.sh"` $COMPOSE up -d supervisorctl_restart php echo -e "${OK} ${GreenBG} 安装完成 ${Font}" echo -e "地址: http://${GreenBG}127.0.0.1:$(env_get APP_PORT)${Font}" echo -e "$res" elif [[ "$1" == "update" ]]; then shift 1 run_mysql backup git fetch --all git reset --hard origin/$(git branch | sed -n -e 's/^\* \(.*\)/\1/p') git pull run_exec php "composer update" run_exec php "php artisan migrate" supervisorctl_restart php $COMPOSE up -d elif [[ "$1" == "uninstall" ]]; then shift 1 read -rp "确定要卸载(含:删除容器、数据库、日志)吗?(y/n): " uninstall [[ -z ${uninstall} ]] && uninstall="N" case $uninstall in [yY][eE][sS] | [yY]) echo -e "${RedBG} 开始卸载... ${Font}" ;; *) echo -e "${GreenBG} 终止卸载。 ${Font}" exit 2 ;; esac $COMPOSE down rm -rf "./docker/mysql/data" rm -rf "./docker/log/supervisor" find "./storage/logs" -name "*.log" | xargs rm -rf echo -e "${OK} ${GreenBG} 卸载完成 ${Font}" elif [[ "$1" == "reinstall" ]]; then shift 1 ./cmd uninstall $@ sleep 3 ./cmd install $@ elif [[ "$1" == "port" ]]; then shift 1 env_set APP_PORT "$1" $COMPOSE up -d echo -e "${OK} ${GreenBG} 修改成功 ${Font}" echo -e "地址: http://${GreenBG}127.0.0.1:$(env_get APP_PORT)${Font}" elif [[ "$1" == "repassword" ]]; then shift 1 run_exec mariadb "sh /etc/mysql/repassword.sh \"$@\"" elif [[ "$1" == "dev" ]] || [[ "$1" == "development" ]]; then shift 1 run_compile dev elif [[ "$1" == "prod" ]] || [[ "$1" == "production" ]]; then shift 1 run_compile prod elif [[ "$1" == "electron" ]]; then shift 1 run_electron $@ elif [[ "$1" == "doc" ]]; then shift 1 run_exec php "php app/Http/Controllers/Api/apidoc.php" docker run -it --rm -v ${cur_path}:/home/node/apidoc kuaifan/apidoc -i app/Http/Controllers/Api -o public/docs elif [[ "$1" == "debug" ]]; then shift 1 if [[ "$@" == "close" ]]; then env_set APP_DEBUG "false" else env_set APP_DEBUG "true" fi supervisorctl_restart php elif [[ "$1" == "https" ]]; then shift 1 if [[ "$@" == "auto" ]]; then env_set APP_SCHEME "auto" else env_set APP_SCHEME "true" fi supervisorctl_restart php elif [[ "$1" == "artisan" ]]; then shift 1 e="php artisan $@" && run_exec php "$e" elif [[ "$1" == "php" ]]; then shift 1 e="php $@" && run_exec php "$e" elif [[ "$1" == "nginx" ]]; then shift 1 e="nginx $@" && run_exec nginx "$e" elif [[ "$1" == "redis" ]]; then shift 1 e="redis $@" && run_exec redis "$e" elif [[ "$1" == "mysql" ]]; then shift 1 if [ "$1" = "backup" ]; then run_mysql backup elif [ "$1" = "recovery" ]; then run_mysql recovery else e="mysql $@" && run_exec mariadb "$e" fi elif [[ "$1" == "composer" ]]; then shift 1 e="composer $@" && run_exec php "$e" elif [[ "$1" == "service" ]]; then shift 1 e="service $@" && run_exec php "$e" elif [[ "$1" == "super" ]] || [[ "$1" == "supervisorctl" ]]; then shift 1 e="supervisorctl $@" && run_exec php "$e" elif [[ "$1" == "models" ]]; then shift 1 run_exec php "php artisan ide-helper:models -W" elif [[ "$1" == "test" ]]; then shift 1 e="./vendor/bin/phpunit $@" && run_exec php "$e" elif [[ "$1" == "restart" ]]; then shift 1 $COMPOSE stop "$@" $COMPOSE start "$@" else $COMPOSE "$@" fi else $COMPOSE ps fi