websoft9/appmanage/api/utils/docker.py

221 lines
7.6 KiB
Python
Raw Normal View History

2023-02-22 09:35:21 +08:00
import os, io, sys, platform, shutil, time, json, datetime
2023-03-25 16:22:07 +08:00
import re, docker, requests
2023-02-22 17:18:24 +08:00
from api.utils import shell_execute
from dotenv import load_dotenv, find_dotenv
2023-02-23 16:19:55 +08:00
import dotenv
2023-02-22 17:18:24 +08:00
from pathlib import Path
2023-03-15 17:55:05 +08:00
from api.utils.common_log import myLogger
2023-04-14 18:36:31 +08:00
from api.utils import shell_execute, const
2023-04-16 14:37:20 +08:00
from api.exception.command_exception import CommandException
2023-04-16 20:44:04 +08:00
from api.service import manage
2023-04-14 19:42:42 +08:00
2023-03-17 09:44:39 +08:00
def pull_images(app_name):
# 备用方法
# 为了防止安装前用户服务器已经有了镜像。导致安装时镜像不重新拉取镜像是老的根据docker-compose.yml 和 .env 获取)
myLogger.info_logger("Pull images complete ...")
2023-03-25 16:22:07 +08:00
2023-03-17 09:44:39 +08:00
def delete_images(app_id):
# 备用方法
# 卸载APP时同时删除dockercompose里面对应的镜像根据docker-compose.yml 和 .env 获取)
myLogger.info_logger("Delete images complete ...")
2023-03-25 16:22:07 +08:00
2023-03-15 10:26:52 +08:00
def get_process_perc(app_name, real_name):
2023-03-30 15:26:22 +08:00
process_now = "pulling"
2023-03-16 11:55:01 +08:00
if if_app_exits(app_name):
2023-03-30 15:26:22 +08:00
process_now = "creating"
2023-03-16 11:55:01 +08:00
2023-03-30 15:26:22 +08:00
if if_app_running(app_name):
process_now = "initing"
if if_app_access(app_name):
2023-04-14 19:42:42 +08:00
process_now = "running"
2023-03-09 11:15:55 +08:00
return process_now
2023-03-09 10:37:49 +08:00
2023-04-14 19:42:42 +08:00
2023-03-30 15:26:22 +08:00
# 已经是running的app怎么知道它已经能够访问如页面能进入如mysql能被客户端连接
def if_app_access(app_name):
return True
2023-04-14 19:42:42 +08:00
2023-03-16 11:55:01 +08:00
def if_app_exits(app_name):
2023-03-25 16:22:07 +08:00
cmd = "docker compose ls -a | grep \'" + app_name + "\\b\'"
2023-03-16 11:55:01 +08:00
output = shell_execute.execute_command_output_all(cmd)
if int(output["code"]) == -1:
return False
else:
return True
2023-03-25 16:22:07 +08:00
2023-04-14 19:42:42 +08:00
2023-03-30 15:26:22 +08:00
def if_app_running(app_name):
cmd = "docker compose ls -a |grep running | grep \'" + app_name + "\\b\'"
output = shell_execute.execute_command_output_all(cmd)
if int(output["code"]) == -1:
return False
else:
return True
2023-04-14 19:42:42 +08:00
2023-04-14 18:36:31 +08:00
def check_appid_exist(app_id):
myLogger.info_logger("Checking check_appid_exist ...")
2023-04-15 16:39:33 +08:00
appList = manage.get_my_app()
find = False
for app in appList:
if app_id == app.app_id:
find = True
break
2023-03-27 16:26:03 +08:00
myLogger.info_logger("Check complete.")
2023-04-15 16:39:33 +08:00
return find
2023-03-27 16:26:03 +08:00
2023-04-15 08:20:29 +08:00
def check_appid_include_rq(app_id):
message = ""
code = None
if app_id == None:
code = const.ERROR_CLIENT_PARAM_BLANK
message = "AppID is null"
2023-04-16 19:41:02 +08:00
elif re.match('^[a-z0-9]+_[a-z0-9]+$', app_id) == None:
2023-04-15 08:20:29 +08:00
code = const.ERROR_CLIENT_PARAM_Format
message = "APP name can only be composed of numbers and lowercase letters"
2023-04-16 20:30:13 +08:00
elif not check_appid_exist(app_id):
2023-04-15 08:20:29 +08:00
code = const.ERROR_CLIENT_PARAM_NOTEXIST
message = "AppID is not exist"
return code, message
2023-04-15 16:39:33 +08:00
2023-04-15 08:20:29 +08:00
def check_app_id(app_id):
message = ""
code = None
if app_id == None:
code = const.ERROR_CLIENT_PARAM_BLANK
message = "AppID is null"
2023-04-16 19:41:24 +08:00
elif re.match('^[a-z0-9]+_[a-z0-9]+$', app_id) == None:
2023-04-15 08:20:29 +08:00
code = const.ERROR_CLIENT_PARAM_Format
message = "APP name can only be composed of numbers and lowercase letters"
2023-04-16 19:05:31 +08:00
myLogger.info_logger(code)
2023-04-14 18:36:31 +08:00
return code, message
2023-03-27 16:26:03 +08:00
def check_vm_resource(app_name):
2023-03-15 17:55:05 +08:00
myLogger.info_logger("Checking virtual memory resource ...")
2023-03-25 17:37:40 +08:00
var_path = "/data/library/apps/" + app_name + "/variables.json"
2023-03-25 16:59:04 +08:00
requirements_var = read_var(var_path, 'requirements')
2023-03-15 14:26:03 +08:00
need_cpu_count = int(requirements_var['cpu'])
2023-03-25 16:22:07 +08:00
cpu_count = int(shell_execute.execute_command_output_all("cat /proc/cpuinfo | grep \'core id\'| wc -l")["result"])
if cpu_count < need_cpu_count:
myLogger.info_logger("Check complete: The number of CPU cores is insufficient!")
2023-03-15 10:53:35 +08:00
return False
2023-03-25 16:22:07 +08:00
need_mem_total = int(requirements_var['memory'])
2023-03-25 18:28:17 +08:00
mem = shell_execute.execute_command_output_all("free -m | grep Mem")["result"].split()
mem_total = float(mem[1]) / 1024
2023-04-01 12:21:36 +08:00
if round(mem_total) < need_mem_total:
2023-03-25 16:22:07 +08:00
myLogger.info_logger("Check complete: The total amount of memory is insufficient!")
return False
2023-03-25 19:13:57 +08:00
mem_free = float(mem[3]) / 1024
2023-04-01 12:21:36 +08:00
if need_mem_total > 4 and round(mem_free) < 4:
2023-03-25 16:22:07 +08:00
myLogger.info_logger("Check complete: There is not enough memory left!")
2023-03-15 10:53:35 +08:00
return False
2023-03-15 14:26:03 +08:00
need_disk = int(requirements_var['disk'])
2023-03-25 19:13:57 +08:00
disk_free = float(
shell_execute.execute_command_output_all("df -m --output=avail /")["result"].split("\n")[1]) / 1024
2023-04-01 12:21:36 +08:00
if round(disk_free) < need_disk - 17:
2023-03-25 16:22:07 +08:00
myLogger.info_logger("Check complete: There are not enough disks left!")
return False
2023-03-25 16:22:07 +08:00
myLogger.info_logger("Check complete.")
2023-03-15 14:07:41 +08:00
return True
2023-03-08 16:33:24 +08:00
2023-03-25 16:22:07 +08:00
2023-04-08 07:59:03 +08:00
def check_app_websoft9(app_name):
2023-03-13 16:32:12 +08:00
# websoft9's support applist
2023-03-15 17:55:05 +08:00
myLogger.info_logger("Checking dir...")
2023-03-25 16:22:07 +08:00
path = "/data/library/apps/" + app_name
2023-03-24 16:41:45 +08:00
is_exists = check_directory(path)
2023-03-21 15:14:35 +08:00
return is_exists
2023-02-22 09:35:21 +08:00
2023-03-25 16:22:07 +08:00
2023-03-24 16:41:45 +08:00
def check_directory(path):
2023-04-16 14:33:53 +08:00
try:
shell_execute.execute_command_output_all("ls " + path)
2023-03-24 16:41:45 +08:00
return True
2023-04-16 14:33:53 +08:00
except CommandException as ce:
2023-03-24 16:41:45 +08:00
return False
2023-03-28 16:07:39 +08:00
def check_app_compose(path):
2023-03-15 17:55:05 +08:00
myLogger.info_logger("Checking port...")
2023-03-10 16:47:11 +08:00
port_dic = read_env(path, "APP_.*_PORT")
2023-03-25 16:22:07 +08:00
# 1.判断/data/apps/app_name/.env中的port是否占用没有被占用方法结束get_start_port方法
2023-03-10 16:47:11 +08:00
for port_name in port_dic:
2023-03-24 15:30:37 +08:00
port_value = get_start_port(port_dic[port_name])
2023-03-10 16:47:11 +08:00
modify_env(path, port_name, port_value)
2023-03-15 17:55:05 +08:00
myLogger.info_logger("Port check complete")
2023-02-23 16:19:55 +08:00
return
2023-03-25 16:22:07 +08:00
2023-03-16 16:49:31 +08:00
def check_app_url(customer_app_name):
myLogger.info_logger("Checking app url...")
2023-03-25 16:22:07 +08:00
2023-03-16 16:49:31 +08:00
# 如果app的.env文件中含有HTTP_URL项目,需要如此设置 HTTP_URL=ip:port
2023-03-16 17:20:11 +08:00
env_path = "/data/apps/" + customer_app_name + "/.env"
if read_env(env_path, "HTTP_URL") != {}:
ip = shell_execute.execute_command_output_all("curl ifconfig.me")["result"]
http_port = list(read_env(path, "APP_HTTP_PORT").values())[0]
url = ip + ":" + http_port
modify_env(path, "HTTP_URL", url)
2023-03-16 16:49:31 +08:00
myLogger.info_logger("App url check complete")
return
2023-03-25 16:22:07 +08:00
def read_env(path, key):
2023-03-15 17:55:05 +08:00
myLogger.info_logger("Read " + path)
2023-03-25 16:22:07 +08:00
output = shell_execute.execute_command_output_all("cat " + path + "|grep " + key)
code = output["code"]
2023-03-10 16:47:11 +08:00
env_dic = {}
if int(code) == 0 and output["result"] != "":
ret = output["result"]
2023-03-10 16:47:11 +08:00
env_list = ret.split()
for env in env_list:
env_dic[env.split("=")[0]] = env.split("=")[1]
2023-03-17 16:31:03 +08:00
myLogger.info_logger("Read " + path + ": " + str(env_dic))
2023-03-10 16:47:11 +08:00
return env_dic
2023-03-25 16:22:07 +08:00
2023-03-09 11:30:22 +08:00
def modify_env(path, env_name, value):
2023-03-15 17:55:05 +08:00
myLogger.info_logger("Modify " + path + "...")
2023-03-24 18:40:31 +08:00
output = shell_execute.execute_command_output_all("sed -n \'/^" + env_name + "/=\' " + path)
if int(output["code"]) == 0 and output["result"] != "":
line_num = output["result"].split("\n")[0]
s = env_name + "=" + value
output = shell_execute.execute_command_output_all("sed -i \'" + line_num + "c " + s + "\' " + path)
if int(output["code"]) == 0:
myLogger.info_logger("Modify " + path + ": Change " + env_name + " to " + value)
2023-03-14 12:57:34 +08:00
2023-03-25 16:22:07 +08:00
2023-03-25 16:59:04 +08:00
def read_var(var_path, var_name):
2023-03-29 14:48:18 +08:00
value = ""
2023-03-15 17:55:05 +08:00
myLogger.info_logger("Read " + var_path)
2023-03-24 18:40:31 +08:00
output = shell_execute.execute_command_output_all("cat " + var_path)
if int(output["code"]) == 0:
var = json.loads(output["result"])
2023-03-14 12:57:34 +08:00
try:
value = var[var_name]
except KeyError:
2023-03-15 17:55:05 +08:00
myLogger.warning_logger("Read " + var_path + ": No key " + var_name)
2023-03-24 18:40:31 +08:00
else:
2023-03-15 17:55:05 +08:00
myLogger.warning_logger(var_path + " not found")
2023-03-14 12:57:34 +08:00
return value
2023-03-24 15:30:37 +08:00
2023-03-25 16:22:07 +08:00
2023-03-24 15:30:37 +08:00
def get_start_port(port):
use_port = port
while True:
cmd = "netstat -ntlp | grep -v only"
output = shell_execute.execute_command_output_all(cmd)
2023-03-25 16:22:07 +08:00
if output["result"].find(use_port) == -1:
2023-03-24 15:30:37 +08:00
break
else:
2023-03-25 16:22:07 +08:00
use_port = str(int(use_port) + 1)
2023-03-24 15:30:37 +08:00
return use_port