| |
| |
| |
| |
| |
| |
| |
| |
| import json |
| import os |
| import re |
| |
| |
| |
| import sys |
| import datetime |
| from typing import Dict, List, Tuple |
|
|
| if "/www/server/panel/class" not in sys.path: |
| sys.path.insert(0, "/www/server/panel/class") |
|
|
| os.chdir("/www/server/panel") |
| import public |
|
|
| if "/www/server/panel" not in sys.path: |
| sys.path.insert(0, "/www/server/panel") |
| from mod.project.nodejs.base import NodeJs |
|
|
|
|
| class main(NodeJs): |
|
|
| def __init__(self): |
| super(main, self).__init__() |
|
|
| |
| def get_pm2_monit(self, get): |
| ''' |
| @name 获取pm2的监控数据 |
| ''' |
| try: |
| self.set_project_model("pm2") |
| get_jlist = self.projectModel.get_jlist() |
| if len(get_jlist) == 0: |
| return public.returnResult(True, '', data=[], code=0) |
|
|
| import pwd |
| data = [] |
| cluster_mode = {} |
|
|
| for jl in get_jlist: |
| if jl.get("name") in ("pm2-sysmonit", "pm2-logrotate"): |
| continue |
|
|
| try: |
| user = pwd.getpwuid(jl.get("pm2_env").get("uid")).pw_name |
| except: |
| user = jl.get("pm2_env").get("USER") |
|
|
| tmp = { |
| "mode": jl.get("pm2_env").get("exec_mode"), |
| "id": jl.get("pm_id"), |
| "pid": jl.get("pid"), |
| "name": jl.get("name"), |
| "user": user, |
| "restart": jl.get("pm2_env").get("restart_time"), |
| "uptime": jl.get("pm2_env").get("pm_uptime"), |
| "cpu": str(jl.get("monit").get("cpu")), |
| "memory": jl.get("monit").get("memory"), |
| "auto_restart": jl.get("pm2_env").get("autorestart"), |
| "status": jl.get("pm2_env").get("status"), |
| "pm2_env": jl.get("pm2_env"), |
| } |
| if jl.get("pm2_env").get("exec_mode") == "fork_mode": |
| if len(cluster_mode) != 0: |
| data.append(cluster_mode) |
| cluster_mode = {} |
| data.append(tmp) |
| elif jl.get("pm2_env").get("exec_mode") == "cluster_mode": |
| if len(cluster_mode) == 0: |
| tmp["mode"] = "fork_mode" |
| cluster_mode["name"] = jl.get("name") |
| cluster_mode["user"] = user |
| cluster_mode["uptime"] = jl.get("pm2_env").get("pm_uptime") |
| cluster_mode["mode"] = jl.get("pm2_env").get("exec_mode") |
| cluster_mode["status"] = jl.get("pm2_env").get("status") |
| cluster_mode["data"] = [] |
| cluster_mode["data"].append(tmp) |
| continue |
|
|
| if cluster_mode.get("name") == jl.get("name") or jl.get("pm2_env").get("NODE_PROJECT_NAME") == cluster_mode.get("name"): |
| tmp["mode"] = "fork_mode" |
| cluster_mode["data"].append(tmp) |
| if jl.get("pm2_env").get("status") == "online" and cluster_mode["status"] != "online": |
| cluster_mode["status"] = "online" |
| else: |
| data.append(cluster_mode) |
| cluster_mode = {} |
| tmp["mode"] = "fork_mode" |
| cluster_mode["name"] = jl.get("name") |
| cluster_mode["user"] = user |
| cluster_mode["uptime"] = jl.get("pm2_env").get("pm_uptime") |
| cluster_mode["mode"] = jl.get("pm2_env").get("exec_mode") |
| cluster_mode["status"] = jl.get("pm2_env").get("status") |
| cluster_mode["data"] = [] |
| cluster_mode["data"].append(tmp) |
|
|
| if len(cluster_mode) != 0: |
| data.append(cluster_mode) |
|
|
| return public.returnResult(True, '', data=data, code=0) |
| except: |
| return public.returnResult(False, '获取pm2监控数据失败', code=5) |
|
|
| def get_pm2_cmd(self, version: str = None): |
| if version is None: |
| import PluginLoader |
| nodejs_version = PluginLoader.plugin_run("nodejs", "get_default_env", None) |
| else: |
| nodejs_version = version |
| if nodejs_version is None: |
| return None |
| args = public.dict_obj() |
| args.nodejs_version = nodejs_version |
| self.set_pm2_cmd(args.nodejs_version) |
|
|
| |
| def get_jlist(self, loop: bool = False): |
| ''' |
| @name 获取pm2的所有进程列表 |
| ''' |
| try: |
| self.get_pm2_cmd() |
| if self.pm2_cmd is None: |
| args = public.dict_obj() |
| nodejs_versions = self.get_nodejs_version(args) |
| if nodejs_versions: |
| nodejs_versions = sorted(nodejs_versions, key=self.version_key, reverse=True) |
| self.get_pm2_cmd(nodejs_versions[0]) |
| else: |
| return [] |
|
|
| if self.pm2_cmd is None: |
| return [] |
|
|
| stdout, stderr = public.ExecShell("export HOME=/root;{} jlist".format(self.pm2_cmd)) |
| if stderr: |
| return [] |
|
|
| return json.loads(stdout) |
| except: |
| try: |
| args = public.dict_obj() |
| nodejs_versions = self.get_nodejs_version(args) |
| if nodejs_versions: |
| nodejs_versions = sorted(nodejs_versions, key=self.version_key, reverse=True) |
| self.get_pm2_cmd(nodejs_versions[0]) |
| else: |
| return [] |
|
|
| if self.pm2_cmd is None: |
| return [] |
|
|
| stdout, stderr = public.ExecShell("export HOME=/root;{} jlist".format(self.pm2_cmd)) |
| if stderr: |
| return [] |
|
|
| return json.loads(stdout) |
| except: |
| return [] |
|
|
| |
| def get_logs(self, mode: str = None, id: str = None, name: str = None, log_type: str = "all", *args, **kwargs): |
| ''' |
| @name 获取指定pm2项目的日志 |
| ''' |
| if log_type != "all": |
| log_type = "--{}".format(log_type) |
| else: |
| log_type = "" |
|
|
| self.get_pm2_cmd() |
| if self.pm2_cmd is None: |
| return public.returnResult(False, '未检测到pm2,请先安装pm2或切换node版本后再试', code=5) |
| if mode == "fork_mode": |
| stdout, stderr = public.ExecShell("sudo {} logs {} --lines 100 {} --nostream".format(self.pm2_cmd, id, log_type)) |
| elif mode == "cluster_mode": |
| stdout, stderr = public.ExecShell( |
| "sudo {} logs {} --lines 100 {} --nostream".format(self.pm2_cmd, name, log_type)) |
| else: |
| return public.returnResult(False, 'mode参数错误', code=5) |
| if stderr and "sudo:" not in stderr: |
| return public.returnResult(False, stderr, code=5) |
| return public.returnResult(True, '', data=stdout, code=0) |
|
|
| |
| def restart(self, mode: str = None, id: str = None, name: str = None, project_name: str = None, *args, **kwargs): |
| ''' |
| @name 重启pm2项目 |
| ''' |
| self.get_pm2_cmd() |
| if self.pm2_cmd is None: |
| return public.returnResult(False, '未检测到pm2,请先安装pm2或切换node版本后再试', code=5) |
|
|
| projects = self.get_jlist() |
| for project in projects: |
| if project.get("name") == name or project.get("pm2_env", {}).get("NODE_PROJECT_NAME") == name: |
| name = project.get("name") |
|
|
| if mode == "fork_mode": |
| stdout, stderr = public.ExecShell("sudo {} restart {}".format(self.pm2_cmd, id)) |
| elif mode == "cluster_mode": |
| stdout, stderr = public.ExecShell("sudo {} restart {}".format(self.pm2_cmd, name)) |
| else: |
| return public.returnResult(False, 'mode参数错误', code=5) |
|
|
| if stderr and "sudo:" not in stderr: |
| return public.returnResult(False, stderr, code=5) |
| return public.returnResult(True, "重启成功", code=0) |
|
|
| |
| def stop(self, mode: str = None, id: str = None, name: str = None, project_name: str = None, *args, **kwargs): |
| ''' |
| @name 停止pm2项目 |
| ''' |
| self.get_pm2_cmd() |
| if self.pm2_cmd is None: |
| return public.returnResult(False, '未检测到pm2,请先安装pm2或切换node版本后再试', code=5) |
| projects = self.get_jlist() |
| for project in projects: |
| if project.get("name") == name or project.get("pm2_env", {}).get("NODE_PROJECT_NAME") == name: |
| name = project.get("name") |
|
|
| if mode == "fork_mode": |
| stdout, stderr = public.ExecShell("sudo {} stop {}".format(self.pm2_cmd, id)) |
| elif mode == "cluster_mode": |
| stdout, stderr = public.ExecShell("sudo {} stop {}".format(self.pm2_cmd, name)) |
| else: |
| return public.returnResult(False, 'mode参数错误', code=5) |
| |
| if stderr and "sudo:" not in stderr: |
| return public.returnResult(False, stderr, code=5) |
| return public.returnResult(True, "停止成功", code=0) |
|
|
| |
| def start(self, mode: str = None, id: str = None, name: str = None, project_name: str = None, *args, **kwargs): |
| ''' |
| @name 启动pm2项目 |
| ''' |
| self.get_pm2_cmd() |
| if self.pm2_cmd is None: |
| return public.returnResult(False, '未检测到pm2,请先安装pm2或切换node版本后再试', code=5) |
|
|
| get = public.dict_obj() |
| get.project_name = project_name |
| return self.start_project(get) |
|
|
| |
| def delete(self, mode: str = None, id: str = None, name: str = None, *args, **kwargs): |
| ''' |
| @name 删除pm2项目 |
| ''' |
| self.get_pm2_cmd() |
| if self.pm2_cmd is None: |
| return public.returnResult(False, '未检测到pm2,请先安装pm2或切换node版本后再试', code=5) |
|
|
| if mode == "fork_mode": |
| stdout, stderr = public.ExecShell("{} delete {}".format(self.pm2_cmd, id)) |
| elif mode == "cluster_mode": |
| stdout, stderr = public.ExecShell("{} delete {}".format(self.pm2_cmd, name)) |
| else: |
| return public.returnResult(False, 'mode参数错误', code=5) |
|
|
| if stderr: |
| return public.returnResult(False, stderr, code=5) |
| return public.returnResult(True, "删除成功", code=0) |
|
|
| |
| def delete_for_ecosystem(self, nodejs_version, config_file): |
| ''' |
| @name 从指定ecosystem.config删除pm2 |
| @param config_file: string 配置文件路径 |
| ''' |
| self.set_pm2_cmd(nodejs_version) |
| if not os.path.exists(config_file): |
| return public.returnResult(False, '指定项目配置文件不存在', code=5) |
| if not os.path.isfile(config_file): |
| return public.returnResult(False, '指定项目配置文件不是一个文件', code=5) |
|
|
| stdout, stderr = public.ExecShell("sudo {} delete {}".format(self.pm2_cmd, config_file)) |
| if stderr and "sudo:" not in stderr: |
| return public.returnResult(False, stderr, code=5) |
| return public.returnResult(True, "删除成功", code=0) |
|
|
| |
| def start_for_ecosystem(self, nodejs_version, config_file): |
| ''' |
| @name 从指定ecosystem.config启动pm2 |
| @param config_file: string 配置文件路径 |
| ''' |
| self.set_pm2_cmd(nodejs_version) |
| if not os.path.exists(config_file): |
| return public.returnResult(False, '指定项目配置文件不存在', code=5) |
| if not os.path.isfile(config_file): |
| return public.returnResult(False, '指定项目配置文件不是一个文件', code=5) |
|
|
| stdout, stderr = public.ExecShell("sudo {} start {}".format(self.pm2_cmd, config_file)) |
| if stderr and "sudo:" not in stderr: |
| return public.returnResult(False, stderr, code=5) |
| return public.returnResult(True, "启动成功", code=0) |
|
|
| |
| def create_project(self, get): |
| ''' |
| @name 创建项目 |
| @param get: dict_obj {} |
| get.project_type string 项目类型 nodejs/pm2/general 必传 |
| get.project_name string 项目名称 my_project 必传 |
| get.nodejs_version string node版本 v20.15.0 必传 |
| get.project_file string 项目启动文件 /www/wwwroot/my_project/server.js 自定义添加时必传 |
| get.project_cwd string 项目路径 /www/wwwroot/my_project 自定义添加时必传 |
| get.cluster int 实例数量 1 必传,默认1 |
| get.max_memory_limit int 最大内存限制 1024 必传,默认1024,mb |
| get.watch 自动重载 bool True/False 必传,默认False |
| get.pkg_manager string 包管理器 none/npm/yarn/pnpm/... 必传,默认none |
| get.not_install_pkg bool 是否安装依赖包 True/False 非必传 |
| get.run_user string 运行用户 www/root/... 必传 |
| get.config_file string 配置文件路径 /www/wwwroot/remix_app/ecosystem.config.cjs 配置文件方式添加时必传 |
| get.config_body string 配置文件内容 非必传 可以单独传config_body,如果选了config_file,这里就是必传,也要传config_body |
| get.port string 端口 4001 非必传 |
| get.release_firewall bool 是否放行防火墙 True/False 非必传 |
| get.is_power_on bool 是否开机启动 True/False 非必传 |
| get.bind_extranet bool 是否绑定外网 True/False 依赖于get.port 非必传 |
| get.domains list 域名列表 ["www.bt.cn", "bt.cn", ...] 非必传 |
| get.project_ps string 备注 ps 非必传 |
| ''' |
| self.set_self_get(get) |
| self.set_def_name(get.def_name) |
| get.config_file = get.get("config_file", "") |
| get.config_body = get.get("config_body", "") |
| get.project_cwd = get.get("project_cwd", "") |
| get.project_file = get.get("project_file", "") |
| get.add_type = get.get("add_type/d", 0) |
|
|
| if get.config_file != "" or get.config_body != "": |
| get.add_type = 1 |
| if get.config_file != "": |
| if not os.path.exists(get.config_file): |
| self.ws_err_exit(False, '{} 指定项目配置文件不存在'.format(get.config_file), code=2) |
| if not os.path.isfile(get.config_file): |
| self.ws_err_exit(False, '{} 指定项目配置文件不是一个文件'.format(get.config_file), code=2) |
| if get.config_body == "": |
| if get.config_file != "": |
| get.config_body = public.readFile(get.config_file) |
| if get.config_body == "": |
| self.ws_err_exit(False, '{} 配置文件格式错误,请检查'.format(get.config_file), code=2) |
| if not "module.exports" in get.config_body and not "apps:" in get.config_body: |
| self.ws_err_exit(False, '{} 配置文件格式错误,请检查'.format(get.config_file), code=2) |
| else: |
| if get.project_file == "": |
| self.ws_err_exit(False, 'project_file参数不能为空', code=2) |
| if not os.path.exists(get.project_file): |
| self.ws_err_exit(False, '{} 指定项目启动文件不存在'.format(get.project_file), code=2) |
| if not os.path.isfile(get.project_file): |
| self.ws_err_exit(False, '{} 指定项目启动文件不是一个文件'.format(get.project_file), code=2) |
|
|
| if get.project_cwd == "": |
| self.ws_err_exit(False, 'project_cwd参数不能为空', code=2) |
| if not os.path.exists(get.project_cwd): |
| self.ws_err_exit(False, '{} 指定项目目录不存在'.format(get.project_cwd), code=2) |
| if not os.path.isdir(get.project_cwd): |
| self.ws_err_exit(False, '{} 指定项目目录不是一个目录'.format(get.project_cwd), code=2) |
|
|
| get.project_name = get.get("project_name", None) |
| get.pm2_name = get.project_name |
| if get.project_name is None: |
| self.ws_err_exit(False, 'project_name参数不能为空', code=2) |
| get.project_name = public.xssencode2(get.project_name) |
| get.project_type = get.get("project_type", None) |
| if get.project_type is None: |
| self.ws_err_exit(False, 'project_type参数不能为空', code=2) |
| if get.project_type != "pm2": |
| self.ws_err_exit(False, '此模型仅支持general项目', code=2) |
| get.run_user = get.get("run_user", "www") |
| get.project_args = get.get("project_args", "") |
| get.env = get.get("env", "") |
| if get.env != "": |
| env = get.env.split("\n") |
| for e in env: |
| if not "=" in e: |
| self.ws_err_exit(False, "环境变量: {} 格式错误,请重新输入例如:key=value".format(e), code=2) |
|
|
| self.set_pm2_cmd(get.nodejs_version) |
| if self.pm2_cmd is None: |
| self.ws_err_exit(False, "请先在【node {}】中安装pm2后再创建".format(get.nodejs_version), code=2) |
|
|
| get.not_install_pkg = get.get("not_install_pkg", False) |
| get.release_firewall = get.get("release_firewall", False) |
| get.is_power_on = get.get("is_power_on", True) |
| get.max_memory_limit = get.get("max_memory_limit", 1024) |
| get.max_memory_restart = "{}M".format(get.max_memory_limit) |
| get.project_ps = get.get("project_ps", "") |
|
|
| get.watch = get.get("watch", False) |
| get.cluster = get.get("cluster/d", 1) |
|
|
| package_file = "{}/package.json".format(get.project_cwd) |
| get.package_info = {} |
| if os.path.exists(package_file): |
| try: |
| get.package_info = json.loads(public.readFile(package_file)) |
| except: |
| pass |
|
|
| if get.package_info: |
| self.check_node_version(get) |
| if not get.not_install_pkg: |
| from mod.project.nodejs.packageManage import PackageManage |
| PackageManage().install_package(get=get, manager=get.pkg_manager, path=get.project_cwd) |
|
|
| if get.config_file != "" and get.config_body != "": |
| |
| if not "module.exports" in get.config_body and not "apps:" in get.config_body: |
| self.ws_err_exit(False, '{} 配置文件格式错误,请检查'.format(get.config_file), code=2) |
|
|
| self.structure_ecosystem_for_config_body(get) |
| else: |
| self.structure_ecosystem(get) |
| |
| project_file_body = public.readFile(get.project_file) |
| if "module.exports" in project_file_body and "apps:" in project_file_body: |
| name_re = re.compile(r'name:.*?,') |
| name = name_re.search(project_file_body).group() |
| get.pm2_name = name.split(":")[1].strip().replace("'", "").replace('"', "").replace(",", "") |
|
|
| get._ws.send(json.dumps(self.wsResult(True, "正在构造启动脚本...", code=0))) |
| self.structure_start_script(get) |
| get._ws.send(json.dumps(self.wsResult(True, "启动脚本构造完成.", code=0))) |
| get._ws.send(json.dumps(self.wsResult(True, "正在写入项目必要配置文件...", code=1))) |
| project_id = self.create_site(get) |
| if project_id is None: |
| self.ws_err_exit(False, '创建网站失败,无法正常写入数据库,请尝试重新添加!', code=2) |
| self.set_config(get.project_name) |
| get._ws.send(json.dumps(self.wsResult(True, "配置文件写入完成.\r\n正在启动项目...", code=1))) |
| start_result = self.start_project(get) |
| if not start_result["status"]: |
| self.ws_err_exit(False, start_result["msg"] if "msg" in start_result else start_result["error_msg"], code=5) |
| get._ws.send(json.dumps(self.wsResult(True, "项目创建成功!", code=-1))) |
| get._ws.close() |
|
|
| |
| def structure_ecosystem_for_config_body(self, get): |
| ''' |
| @name 从config_body构造 ecosystem |
| ''' |
| log_path = os.path.join(self.pm2_logs_path, get.project_name) |
| if not os.path.exists(log_path): |
| os.makedirs(log_path, 493, True) |
| out_file = os.path.join(str(log_path), "out.log") |
| if os.path.exists(out_file) and os.path.isfile(out_file): |
| public.writeFile(out_file, "") |
| err_file = os.path.join(str(log_path), "err.log") |
| if os.path.exists(err_file) and os.path.isfile(err_file): |
| public.writeFile(err_file, "") |
|
|
| public.ExecShell("chown -R {user}:{user} {project_cwd}".format(user=get.run_user, project_cwd=log_path)) |
| public.ExecShell("chmod 755 -R {}".format(log_path)) |
| public.set_own(out_file, get.run_user, get.run_user) |
| public.set_mode(out_file, 755) |
| public.set_own(err_file, get.run_user, get.run_user) |
| public.set_mode(err_file, 755) |
|
|
| user_field = "user: '{}',".format(get.run_user) |
| cwd_field = 'cwd: "{}",'.format(get.project_cwd) |
| out_file_field = 'out_file: "{}",'.format(out_file) |
| error_file_field = 'error_file: "{}",'.format(err_file) |
| log_date_format_field = 'log_date_format: "YYYY-MM-DD HH:mm:ss",' |
| merge_logs_field = 'merge_logs: true,' |
|
|
| name_re = re.compile(r'name:.*?,') |
| name = name_re.search(get.config_body).group() |
| get.pm2_name = name.split(":")[1].strip().replace("'", "").replace('"', "").replace(",", "") |
|
|
| user_re = re.compile(r'user:.*?,') |
| cwd_re = re.compile(r'cwd:.*?,') |
| out_file_re = re.compile(r'out_file:.*?,') |
| error_file_re = re.compile(r'error_file:.*?,') |
| log_date_format_re = re.compile(r'log_date_format:.*?,') |
| merge_logs_re = re.compile(r'merge_logs:.*?,') |
|
|
| get.config_body = self.replace_or_add_field(get.config_body, user_re, user_field) |
| get.config_body = self.replace_or_add_field(get.config_body, cwd_re, cwd_field) |
| get.config_body = self.replace_or_add_field(get.config_body, out_file_re, out_file_field) |
| get.config_body = self.replace_or_add_field(get.config_body, error_file_re, error_file_field) |
| get.config_body = self.replace_or_add_field(get.config_body, log_date_format_re, log_date_format_field) |
| get.ecosystem = self.replace_or_add_field(get.config_body, merge_logs_re, merge_logs_field) |
|
|
| |
| def structure_ecosystem(self, get): |
| ''' |
| @name 构造 ecosystem |
| ''' |
| log_path = os.path.join(self.pm2_logs_path, get.project_name) |
| if not os.path.exists(log_path): |
| os.makedirs(log_path, 493, True) |
| out_file = os.path.join(str(log_path), "out.log") |
| if os.path.exists(out_file) and os.path.isfile(out_file): |
| public.writeFile(out_file, "") |
| err_file = os.path.join(str(log_path), "err.log") |
| if os.path.exists(err_file) and os.path.isfile(err_file): |
| public.writeFile(err_file, "") |
|
|
| public.ExecShell("chown -R {user}:{user} {project_cwd}".format(user=get.run_user, project_cwd=log_path)) |
| public.ExecShell("chmod 755 -R {}".format(log_path)) |
| public.set_own(out_file, get.run_user, get.run_user) |
| public.set_mode(out_file, 755) |
| public.set_own(err_file, get.run_user, get.run_user) |
| public.set_mode(err_file, 755) |
|
|
| exec_mode = "\"fork\"," |
| instances = "" |
| if get.cluster > 1: |
| exec_mode = "\"cluster\"," |
| instances = "\n instances: \"{instances}\",".format(instances=get.cluster) |
|
|
| env = "" |
| if get.env != "": |
| env = "\n env: {env},".format(env=json.dumps(dict([i.split("=") for i in get.env.split("\n")]))) |
| get.ecosystem = '''module.exports = {{ |
| apps: [ |
| {{ |
| name: "{project_name}", |
| namespace: "{project_name}", |
| max_memory_restart: "{max_memory_restart}", |
| user: "{run_user}", |
| exec_mode: {exec_mode}{instances}{env} |
| cwd: "{project_cwd}", |
| script: "{project_file}", |
| args: "{project_args}", |
| watch: {watch}, |
| out_file: {out_file}, |
| error_file: {error_file}, |
| log_date_format: "YYYY-MM-DD HH:mm:ss", |
| merge_logs: true, |
| interpreter: "{project_node}", |
| }} |
| ] |
| }}'''.format( |
| project_name=get.project_name, |
| max_memory_restart=get.max_memory_restart, |
| run_user=get.run_user, |
| exec_mode=exec_mode, |
| instances=instances, |
| env=env, |
| project_cwd=get.project_cwd, |
| project_file=get.project_file, |
| project_args=get.project_args, |
| watch="true" if get.watch else "false", |
| out_file="\"{}/out.log\"".format(log_path), |
| error_file="\"{}/err.log\"".format(log_path), |
| project_node="/www/server/nodejs/{}/bin/node".format(get.nodejs_version), |
| ) |
|
|
| |
| def structure_start_script(self, get): |
| ''' |
| @name 构造传统项目的启动脚本 |
| ''' |
| last_env = self.get_last_env(get.nodejs_version) |
| command = '''{last_env} |
| export NODE_PROJECT_NAME="{project_name}" |
| export HOME=/root |
| cd {project_cwd} |
| {pm2_cmd} start {ecosystem_config_file}'''.format( |
| last_env=last_env, |
| project_name=get.project_name, |
| project_cwd=get.project_cwd, |
| ecosystem_config_file=get.project_file, |
| pm2_cmd=self.pm2_cmd, |
| ) |
|
|
| get.config_file = get.project_file |
| get.config_body = get.ecosystem |
| get.project_script = command |
|
|
| |
| def start_project(self, get): |
| ''' |
| @name 启动项目 |
| ''' |
| |
| self.stop_project(get) |
|
|
| project_find = self.get_project_find(get.project_name) |
| if not project_find: return public.returnResult(False, "项目不存在", code=5) |
|
|
| project_find = self.get_project_find(get.project_name) |
| if project_find['edate'] != "0000-00-00" and project_find['edate'] < datetime.datetime.today().strftime("%Y-%m-%d"): |
| return public.returnResult(False, "当前项目已过期,请重新设置项目到期时间", code=5) |
|
|
| self._update_project(get.project_name, project_find) |
| if not os.path.exists(project_find['path']): |
| error_msg = '启动失败,Nodejs项目{},运行目录{}不存在!'.format(get.project_name, project_find['path']) |
| public.WriteLog(self.log_name, error_msg) |
| return public.returnResult(False, error_msg, code=5) |
|
|
| |
| project_script = project_find['project_config']['project_script'].strip().replace(' ', ' ') |
| log_file = "{}/{}.log".format(project_find['project_config']["log_path"], project_find["name"]) |
| if not project_script: return public.returnResult(True, "未配置启动脚本") |
|
|
| public.writeFile(log_file, '') |
| public.ExecShell('chmod 777 {}'.format(log_file)) |
| |
| start_cmd = project_script |
| script_file = "{}/{}.sh".format(self.node_run_scripts, get.project_name) |
| public.writeFile(script_file, start_cmd) |
|
|
| |
| stdout, stderr = public.ExecShell("bash {}".format(script_file), env=os.environ.copy()) |
| public.writeFile(log_file, stdout + stderr) |
|
|
| if stderr: |
| return public.returnResult(False, '启动失败: {}'.format(stderr), code=5) |
|
|
| self.start_by_user(project_find["id"]) |
| return public.returnResult(True, '启动成功') |
|
|
| |
| def stop_project(self, get): |
| ''' |
| @name 停止项目 |
| @author hwliang<2021-08-09> |
| @param get<dict_obj>{ |
| project_name: string<项目名称> |
| } |
| @return dict |
| ''' |
| project_find = self.get_project_find(get.project_name) |
| if not project_find: return public.returnResult(False, '项目不存在', code=5) |
| if project_find['edate'] != "0000-00-00" and project_find['edate'] < datetime.datetime.today().strftime("%Y-%m-%d"): |
| return public.returnResult(False, '当前项目已过期,请重新设置项目到期时间', code=5) |
| project_script = project_find['project_config']['project_script'].strip().replace(' ', ' ') |
| project_script = project_script.replace('pm2 start', 'pm2 stop') |
| stdout, stderr = public.ExecShell(project_script) |
| public.writeFile("{}/{}.log".format(project_find['project_config']["log_path"], project_find["name"]), |
| stdout + stderr) |
| if stderr: |
| return public.returnResult(False, '停止失败: {}'.format(stderr), code=5) |
| return public.returnResult(True, '停止成功') |
|
|
| |
| def restart_project(self, get): |
| ''' |
| @name 重启项目 |
| @author hwliang<2021-08-09> |
| @param get<dict_obj>{ |
| project_name: string<项目名称> |
| } |
| @return |
| ''' |
| project_find = self.get_project_find(get.project_name) |
| if project_find: |
| if project_find['edate'] != "0000-00-00" and project_find['edate'] < datetime.datetime.today().strftime("%Y-%m-%d"): |
| return public.returnResult(False, '当前项目已过期,请重新设置项目到期时间', code=5) |
| res = self.stop_project(get) |
| if not res['status']: return res |
| res = self.start_project(get) |
| if not res['status']: return res |
| return public.returnResult(True, '重启成功') |
|
|