apiController.java 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363
  1. package com.yyandywt99.pandoraNext.controller;
  2. import com.github.dockerjava.api.DockerClient;
  3. import com.github.dockerjava.api.command.InspectContainerResponse;
  4. import com.github.dockerjava.core.DockerClientBuilder;
  5. import com.yyandywt99.pandoraNext.anno.Log;
  6. import com.yyandywt99.pandoraNext.pojo.Result;
  7. import com.yyandywt99.pandoraNext.pojo.token;
  8. import com.yyandywt99.pandoraNext.service.systemService;
  9. import lombok.extern.slf4j.Slf4j;
  10. import org.springframework.beans.factory.annotation.Autowired;
  11. import org.springframework.beans.factory.annotation.Value;
  12. import org.springframework.web.bind.annotation.*;
  13. import javax.servlet.http.HttpServletRequest;
  14. import java.util.List;
  15. /**
  16. * @author Yangyang
  17. * @create 2023-11-07 14:55
  18. */
  19. @Slf4j
  20. @RestController()
  21. @RequestMapping("/api")
  22. public class apiController {
  23. @Autowired
  24. private com.yyandywt99.pandoraNext.service.apiService apiService;
  25. @Value("${deployPosition}")
  26. private String deployPosition;
  27. public String deploy = "default";
  28. /**
  29. * @param name
  30. * @return 通过name获取到(tokens.json)文件里的全部值
  31. */
  32. @GetMapping("seleteToken")
  33. public Result seleteToken(@RequestParam("name") String name){
  34. try {
  35. List<token> res = apiService.seleteToken(name);
  36. return Result.success(res);
  37. } catch (Exception e) {
  38. e.printStackTrace();
  39. return Result.error("获取失败");
  40. }
  41. }
  42. /**
  43. * @param token
  44. * @return 添加token和其余变量到指定文件(tokens.json)
  45. */
  46. @Log
  47. @PostMapping("addToken")
  48. public Result addToken(@RequestBody token token){
  49. try {
  50. String res = apiService.addToken(token);
  51. if(res.length() > 300){
  52. return Result.success(res);
  53. }
  54. else if(res.length() == 0){
  55. return Result.success("添加成功,已装填你的token");
  56. }
  57. else{
  58. return Result.error(res);
  59. }
  60. } catch (Exception e) {
  61. e.printStackTrace();
  62. return Result.error("添加失败");
  63. }
  64. }
  65. /**
  66. * @param token
  67. * @return 通过传入token,修改(tokens.json)文件里的值
  68. */
  69. @Log
  70. @PostMapping("requiredToken")
  71. public Result requiredToken(@RequestBody token token){
  72. try {
  73. String res = apiService.requiredToken(token);
  74. if(res.equals("修改成功!")){
  75. return Result.success(res);
  76. }
  77. else{
  78. return Result.error(res);
  79. }
  80. } catch (Exception e) {
  81. e.printStackTrace();
  82. return Result.error("修改token失败");
  83. }
  84. }
  85. /**
  86. * @param name
  87. * @return 通过token用户名,删除(tokens.json)文件里的值
  88. */
  89. @Log
  90. @PutMapping("deleteToken")
  91. public Result deleteToken(@RequestParam String name){
  92. try {
  93. String res = apiService.deleteToken(name);
  94. log.info(res);
  95. return Result.success(res);
  96. } catch (Exception e) {
  97. e.printStackTrace();
  98. return Result.error("删除失败");
  99. }
  100. }
  101. @Value("${deployWay}")
  102. private String deployWay;
  103. /**
  104. * @return 通过访问restart,重启PandoraNext服务
  105. */
  106. @GetMapping("/restart")
  107. public Result restartContainer() {
  108. try {
  109. restartContainer("PandoraNext");
  110. return Result.success("重启PandoraNext镜像成功");
  111. } catch (Exception e) {
  112. log.error("重启PandoraNext镜像失败!", e);
  113. return Result.error("重启PandoraNext镜像失败!");
  114. }
  115. }
  116. /**
  117. * @return 通过访问close,关闭PandoraNext服务
  118. */
  119. @GetMapping("/close")
  120. public Result closeContainer() {
  121. String containerName = "PandoraNext";
  122. if (deployWay.contains("docker")) {
  123. docker(containerName,"pause");
  124. return Result.success("暂停PandoraNext镜像成功");
  125. }
  126. else if (deployWay.equals("releases")) {
  127. try {
  128. closeRelease(containerName);
  129. return Result.success("暂停PandoraNext镜像成功");
  130. } catch (Exception e) {
  131. throw new RuntimeException(e);
  132. }
  133. }
  134. else {
  135. return Result.success("jar用错名称");
  136. }
  137. }
  138. private static boolean isContainerPaused(DockerClient dockerClient, String containerIdOrName) {
  139. // 使用 Docker Java API 查询容器信息
  140. InspectContainerResponse containerInfo = dockerClient.inspectContainerCmd(containerIdOrName).exec();
  141. // 获取容器状态
  142. return containerInfo.getState().getPaused();
  143. }
  144. /**
  145. * @return 通过访问open,开启PandoraNext服务
  146. */
  147. @GetMapping("/open")
  148. public Result openContainer(){
  149. // 要检查的容器ID或名称
  150. String containerName = "PandoraNext";
  151. if (deployWay.contains("docker")) {
  152. try {
  153. // Docker 客户端初始化
  154. DockerClient dockerClient = DockerClientBuilder.getInstance().build();
  155. // 检查容器状态
  156. boolean isPaused = isContainerPaused(dockerClient, containerName);
  157. if (isPaused) {
  158. log.info("容器 " + containerName + " 已暂停。");
  159. docker(containerName,"unpause");
  160. return Result.success("开启PandoraNext镜像成功");
  161. }
  162. // 关闭 Docker 客户端连接
  163. dockerClient.close();
  164. return Result.success("容器 " + containerName + " 未暂停,不能重复启动");
  165. } catch (Exception e) {
  166. throw new RuntimeException(e);
  167. }
  168. }
  169. else if (deployWay.equals("releases")) {
  170. try {
  171. openRelease(containerName);
  172. return Result.success("开启PandoraNext镜像成功");
  173. } catch (Exception e) {
  174. throw new RuntimeException(e);
  175. }
  176. }
  177. else {
  178. return Result.success("jar用错名称");
  179. }
  180. }
  181. @Autowired
  182. private systemService systemService;
  183. @Autowired
  184. private HttpServletRequest httpServletRequest;
  185. /**
  186. * @pandoraNext端口号
  187. */
  188. @Value("${pandoraNext_Url}")
  189. private String pandoraNext_Url;
  190. /**
  191. * @return 通过访问open,重载PandoraNext服务
  192. */
  193. @GetMapping("/reload")
  194. public Result reloadContainer(){
  195. try {
  196. String baseUrlWithoutPath = pandoraNext_Url;
  197. log.info(baseUrlWithoutPath);
  198. String setup_password = systemService.selectSetting().getSetup_password();
  199. String reloadCommand = "curl -H \"Authorization: Bearer "
  200. + setup_password + "\" -X POST \"" + baseUrlWithoutPath + "/setup/reload\"";
  201. // 执行重载进程的命令
  202. Process reloadProcess = executeCommand(reloadCommand);
  203. log.info("重载命令:"+reloadCommand);
  204. // 等待重载进程完成
  205. try {
  206. int exitCode = reloadProcess.waitFor();
  207. if (exitCode != 0) {
  208. log.info("无法重载PandoraNext服务");
  209. return Result.success("无法重载PandoraNext服务");
  210. }
  211. return Result.success("重置PandoraNext服务成功!");
  212. } catch (Exception e) {
  213. throw new RuntimeException(e);
  214. }
  215. } catch (Exception e) {
  216. e.printStackTrace();
  217. }
  218. return Result.success("无法重载PandoraNext服务");
  219. }
  220. /**
  221. * containerName
  222. * 重启containerName的容器
  223. * 分为docker和releases
  224. */
  225. public void restartContainer(String containerName){
  226. log.info(deployWay);
  227. if (deployWay.contains("docker")) {
  228. docker(containerName,"restart");
  229. }
  230. else if (deployWay.equals("releases")) {
  231. try {
  232. try {
  233. //先确保是开启状态
  234. openRelease(containerName);
  235. //再关闭
  236. Thread.sleep(500);
  237. closeRelease(containerName);
  238. //在重启
  239. Thread.sleep(500);
  240. openRelease(containerName);
  241. } catch (Exception e) {
  242. throw new RuntimeException(e);
  243. }
  244. } catch (Exception e) {
  245. log.info("无法重启PandoraNext服务");
  246. throw new RuntimeException(e);
  247. }
  248. }
  249. else {
  250. log.info("jar包填错信息");
  251. }
  252. }
  253. /**
  254. * releases命令
  255. * containeeName:容器名
  256. * 关闭容器项目
  257. */
  258. public void closeRelease(String containName){
  259. try {
  260. String killCommand = "pkill " + containName;
  261. log.info(killCommand);
  262. int exitCode = 0;
  263. try {
  264. // 执行杀死进程的命令
  265. Process killProcess = executeCommand(killCommand);
  266. // 等待杀死进程完成
  267. try {
  268. exitCode = killProcess.waitFor();
  269. } catch (Exception e) {
  270. throw new RuntimeException(e);
  271. }
  272. } catch (Exception e) {
  273. throw new RuntimeException(e);
  274. }
  275. if (exitCode != 0) {
  276. log.info("无法关闭PandoraNext服务");
  277. throw new RuntimeException("无法关闭PandoraNext服务");
  278. }
  279. log.info("关闭PandoraNext服务成功!");
  280. } catch (Exception e) {
  281. throw new RuntimeException();
  282. }
  283. }
  284. /**
  285. * releaser命令
  286. * containeeName:容器名
  287. * 开启容器项目
  288. */
  289. public void openRelease(String containerName){
  290. try {
  291. String projectRoot;
  292. if(deploy.equals(deployPosition)){
  293. projectRoot = System.getProperty("user.dir");
  294. log.info(projectRoot);
  295. }
  296. else{
  297. projectRoot = deployPosition;
  298. }
  299. String startCommand = "cd " + projectRoot + " && nohup ./" + containerName + " > output.log 2>&1 & echo $! > pid.txt";
  300. log.info(startCommand);
  301. Process startProcess = executeCommand(startCommand);
  302. int exitCode = startProcess.waitFor();
  303. if (exitCode != 0) {
  304. log.info("无法启动PandoraNext服务");
  305. throw new RuntimeException("无法启动PandoraNext服务");
  306. }
  307. log.info("启动PandoraNext服务成功!");
  308. } catch (Exception e) {
  309. throw new RuntimeException();
  310. }
  311. }
  312. /**
  313. * docker命令
  314. * containeeName:容器名
  315. * way:命令方法(启动:start 暂停:pause 重启:restart)
  316. */
  317. public void docker(String containerName,String way){
  318. try {
  319. String dockerCommand = "docker "+ way + " " + containerName;
  320. Process process = executeCommand(dockerCommand);
  321. int exitCode = process.waitFor();
  322. if (exitCode != 0) {
  323. log.info("无法"+way+"PandoraNext服务");
  324. throw new RuntimeException("无法"+way+"PandoraNext服务");
  325. }
  326. log.info(way+"PandoraNext服务");
  327. } catch (Exception e) {
  328. throw new RuntimeException();
  329. }
  330. }
  331. /**
  332. * release 命令函数
  333. * command:命令
  334. * 用 bash ,-c ,来包裹命令增加其稳定性
  335. */
  336. public Process executeCommand(String command){
  337. try {
  338. ProcessBuilder processBuilder = new ProcessBuilder("bash", "-c", command);
  339. return processBuilder.start();
  340. } catch (Exception e) {
  341. throw new RuntimeException();
  342. }
  343. }
  344. }