348 lines
16 KiB
Java
348 lines
16 KiB
Java
package com.educoder.bridge.controller;
|
||
|
||
import com.alibaba.fastjson.JSONObject;
|
||
import com.educoder.bridge.exception.GameException;
|
||
import com.educoder.bridge.service.*;
|
||
import com.educoder.bridge.settings.AppConfig;
|
||
import com.educoder.bridge.utils.*;
|
||
import io.swagger.annotations.Api;
|
||
import io.swagger.annotations.ApiOperation;
|
||
import io.swagger.annotations.ApiParam;
|
||
import org.apache.commons.lang.StringUtils;
|
||
import org.slf4j.Logger;
|
||
import org.slf4j.LoggerFactory;
|
||
import org.springframework.beans.factory.annotation.Autowired;
|
||
import org.springframework.http.MediaType;
|
||
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;
|
||
import org.springframework.web.bind.annotation.RequestMapping;
|
||
import org.springframework.web.bind.annotation.RequestMethod;
|
||
import org.springframework.web.bind.annotation.RequestParam;
|
||
import org.springframework.web.bind.annotation.RestController;
|
||
|
||
import java.io.File;
|
||
import java.text.SimpleDateFormat;
|
||
|
||
/**
|
||
* 和实训相关的接口
|
||
*
|
||
* @author weishao
|
||
* @date 2017/11/02
|
||
*/
|
||
@Api(value = "game控制器", hidden = true)
|
||
@RestController
|
||
@RequestMapping("/game")
|
||
public class GameController extends BaseController {
|
||
@Autowired
|
||
private AppConfig appConfig;
|
||
@Autowired
|
||
private GameService gameService;
|
||
@Autowired
|
||
private KubernetesService kubernetesService;
|
||
@Autowired
|
||
private K8sService k8sService;
|
||
@Autowired
|
||
private ThreadPoolTaskExecutor threadPoolTaskExecutor;
|
||
|
||
private final static Logger logger = LoggerFactory.getLogger(GameController.class);
|
||
|
||
/**
|
||
* 开启实训:
|
||
* 克隆版本库
|
||
*
|
||
* @return
|
||
*/
|
||
@RequestMapping(path = "/openGameInstance")
|
||
@ApiOperation(value = "开启实训", httpMethod = "POST", produces = MediaType.APPLICATION_FORM_URLENCODED_VALUE)
|
||
public JSONObject openGameInstance(
|
||
@ApiParam(name = "tpmGitURL", required = true, value = "Tpm的gitUrl,需要base64编码") @RequestParam String tpmGitURL,
|
||
@ApiParam(name = "tpiID", required = true, value = "实训实例的ID") @RequestParam String tpiID,
|
||
@ApiParam(name = "tpiRepoName", required = true, value = "tpiRepoName") @RequestParam String tpiRepoName
|
||
) throws Exception {
|
||
logger.info("tpmGitURL: {}\n, tpiID: {}, tpiRepoName: {}", tpmGitURL, tpiID, tpiRepoName);
|
||
|
||
JSONObject response = new JSONObject();
|
||
|
||
// 设定工作路径为${workspace}/myshixun_${tpiID}
|
||
String path = appConfig.getWorkspace() + File.separator + "myshixun_" + tpiID;
|
||
|
||
// 对当前TPI,从TPM clone 版本库
|
||
tpmGitURL = Base64Util.decode(tpmGitURL);
|
||
gameService.gitClone(path, tpmGitURL, "remote_origin", tpiRepoName);
|
||
|
||
response.put("code", 0);
|
||
response.put("msg", "开启成功");
|
||
return response;
|
||
}
|
||
|
||
/**
|
||
* 评测:
|
||
* 对每一次评测请求,先判断是否能立即执行还是需要先排队,如果能立即执行,开启一个线程执行以下步骤
|
||
* 如果需要排队,将相关参数封装成一个线程,存入redis队列,等待执行,并返回标志位给前台,通知前台进行轮询
|
||
*
|
||
* 1. gitpull
|
||
* 2. 如果有端口映射
|
||
* 3. 创建pod 和 service 并 创建定时删除pod的定时任务
|
||
* 4. 在pod中执行评测脚本
|
||
* 5. 回传结果
|
||
*
|
||
*
|
||
* @param tpiID
|
||
* @param tpiGitURL
|
||
* @param buildID
|
||
* @param instanceChallenge
|
||
* @param resubmit
|
||
* @param times
|
||
* @return
|
||
* @throws Exception
|
||
*/
|
||
@RequestMapping(path = "/gameEvaluate")
|
||
@ApiOperation(value = "评测", httpMethod = "POST", produces = MediaType.APPLICATION_FORM_URLENCODED_VALUE)
|
||
public JSONObject gameEvaluate(
|
||
@ApiParam(name = "tpiID", required = true, value = "实训实例的ID") @RequestParam String tpiID,
|
||
@ApiParam(name = "tpiGitURL", required = true, value = "学员对应当前实训的版本库地址,需要base64编码") @RequestParam String tpiGitURL,
|
||
@ApiParam(name = "buildID", required = true, value = "本次评测ID") @RequestParam String buildID,
|
||
@ApiParam(name = "instanceChallenge", required = true, value = "当前处在第几关") @RequestParam String instanceChallenge,
|
||
@ApiParam(name = "testCases", required = true, value = "测试用例") @RequestParam String testCases,
|
||
@ApiParam(name = "tpmScript", required = true, value = "tpm评测脚本,需要base64编码") @RequestParam String tpmScript,
|
||
@ApiParam(name = "timeLimit", required = false, value = "时间限制") Integer timeLimit,
|
||
@ApiParam(name = "resubmit", required = true, value = "是否是重复评测") @RequestParam String resubmit,
|
||
@ApiParam(name = "times", required = true, value = "是否是时间轮询请求") @RequestParam Integer times,
|
||
@ApiParam(name = "needPortMapping", required = false, value = "容器中需要被映射的端口") Integer needPortMapping,
|
||
@ApiParam(name = "podType", required = true, value = "pod类型(0.evaluate,1.webssh,2.evassh)") @RequestParam Integer podType,
|
||
@ApiParam(name = "file", required = false, value = "需要传文件的实训,给出文件存放路径(一个目录)及文件类型") String file,
|
||
@ApiParam(name = "containers", required = true, value = "需要使用的容器,base64编码") @RequestParam String containers)
|
||
throws Exception {
|
||
// 记录开始时间
|
||
SimpleDateFormat dateformat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss:SSS");
|
||
logger.info("training_task_status start#1**{}**** {}", tpiID, dateformat.format(System.currentTimeMillis()));
|
||
|
||
long evaStart = System.currentTimeMillis();
|
||
String evaluateStart = dateformat.format(evaStart);
|
||
JSONObject cost = new JSONObject();
|
||
cost.put("evaluateStart", evaluateStart);
|
||
cost.put("evaluateAllTime", evaStart);
|
||
ConstantUtil.costMap.put(tpiID, cost);
|
||
|
||
JSONObject response = new JSONObject();
|
||
|
||
// 所有构建需要的参数放到map中,供执行线程使用
|
||
tpiGitURL = Base64Util.decode(tpiGitURL);
|
||
needPortMapping = needPortMapping == null ? 0 : needPortMapping;
|
||
timeLimit = timeLimit == null ? Integer.parseInt(appConfig.getDefaultTimeLimit()) : timeLimit;
|
||
testCases = Base64Util.decode(testCases);
|
||
containers = Base64Util.decode(containers);
|
||
|
||
// 每次评测均生成新TPI评测脚本
|
||
tpmScript = Base64Util.decode(tpmScript);
|
||
String path = appConfig.getWorkspace() + File.separator + "myshixun_" + tpiID;
|
||
String tpiRepoName = StringHelper.getRepoName(tpiGitURL);
|
||
logger.info("generateTpiEvaluateShellScript start#2**{}**** {}", tpiID, dateformat.format(System.currentTimeMillis()));
|
||
gameService.generateTpiEvaluateShellScript(tpmScript, path, tpiRepoName);
|
||
|
||
logger.info("retryformat start#3**{}**** {}", tpiID, dateformat.format(System.currentTimeMillis()));
|
||
JSONObject buildParams = new JSONObject(true);
|
||
buildParams.put("tpiID", tpiID);
|
||
buildParams.put("tpiGitURL", tpiGitURL);
|
||
buildParams.put("buildID", buildID);
|
||
buildParams.put("instanceChallenge", instanceChallenge);
|
||
buildParams.put("testCases", testCases);
|
||
buildParams.put("timeLimit", timeLimit);
|
||
buildParams.put("resubmit", resubmit);
|
||
buildParams.put("needPortMapping", needPortMapping);
|
||
buildParams.put("podType", podType);
|
||
buildParams.put("containers", containers);
|
||
|
||
// 若实训生成文件
|
||
if (!StringUtils.isEmpty(file)) {
|
||
file = Base64Util.decode(file);
|
||
buildParams.put("file", file);
|
||
JSONObject jsonObject = JSONObject.parseObject(file);
|
||
// 清空目标文件夹以防止影响此次评测结果
|
||
String dir = path + "/" + tpiRepoName + "/" + jsonObject.getString("path");
|
||
StringHelper.deleteFiles(dir, "pic");
|
||
StringHelper.deleteFiles(dir, "apk");
|
||
StringHelper.deleteFiles(dir, "html");
|
||
}
|
||
|
||
// 最大running pod数量
|
||
int maxRunningPodNum = Integer.parseInt(appConfig.getMaxRunningPodNum());
|
||
|
||
// podName
|
||
String podName = podType == 0 ? "evaluate-" + tpiID : "evassh-" + tpiID;
|
||
|
||
// 若需要端口映射服务,则分配端口,将podName-port键值对存储于redis
|
||
String port = "-1";
|
||
if(needPortMapping != -1) {
|
||
port = JedisUtil.hget("port", podName);
|
||
if(port == null) {
|
||
port = PortUtil.getPort() + "";
|
||
JedisUtil.hset("port", podName, port);
|
||
}
|
||
}
|
||
|
||
// 构建任务字符串,便于redis存储
|
||
String task = buildParams.toJSONString();
|
||
|
||
// 此次请求只为轮询时间,只返回轮询的时间结果
|
||
if (times != 1) {
|
||
double buildRank;
|
||
|
||
try {
|
||
buildRank = JedisUtil.zrank("task", task);
|
||
} catch (Exception e) {
|
||
response.put("ableToCreate", 0);
|
||
response.put("waitNum", maxRunningPodNum + JedisUtil.zlen("task"));
|
||
response.put("code", 0);
|
||
response.put("msg", "等待评测");
|
||
return response;
|
||
}
|
||
|
||
if (buildRank != Double.MIN_VALUE) {
|
||
logger.debug("任务:{} : 还在等待队列中!", task);
|
||
|
||
response.put("ableToCreate", 0);
|
||
response.put("waitNum", buildRank == Double.MAX_VALUE ? maxRunningPodNum + JedisUtil.zlen("task") : buildRank + maxRunningPodNum);
|
||
response.put("code", 0);
|
||
response.put("msg", "等待评测");
|
||
return response;
|
||
} else {
|
||
// 轮询时间请求,发现目标任务不在队列,就直接认为其正在运行
|
||
logger.debug("任务:{} : 正在执行中!", task);
|
||
|
||
response.put("ableToCreate", 1);
|
||
response.put("waitNum", 0);
|
||
response.put("code", 0);
|
||
response.put("port", Integer.parseInt(port));
|
||
response.put("msg", "正在评测");
|
||
return response;
|
||
}
|
||
}
|
||
|
||
|
||
logger.info("kubernetes start#4**{}**** {}", tpiID, dateformat.format(System.currentTimeMillis()));
|
||
// 获取Running状态的pod数
|
||
int runningPodNum = kubernetesService.getRunningPodNum();
|
||
// 判断是否可以立即执行(若pod已经存在,或是,pod不存在但是此时可以创建pod)
|
||
boolean executeImmediately = kubernetesService.isPodRunning("tpiID", tpiID)
|
||
|| (runningPodNum < maxRunningPodNum && k8sService.ableToEvaluate());
|
||
|
||
if (executeImmediately) {
|
||
// 直接执行任务
|
||
logger.info("execute start#5**{}**** {}", tpiID, dateformat.format(System.currentTimeMillis()));
|
||
BuildThread buildThread = gameService.getBuildThread(buildParams);
|
||
threadPoolTaskExecutor.execute(buildThread);
|
||
long evaEnd = System.currentTimeMillis();
|
||
long costTime = evaEnd - evaStart;
|
||
|
||
response.put("ableToCreate", 1);
|
||
response.put("costTime", costTime);
|
||
response.put("waitNum", 0);
|
||
response.put("code", 0);
|
||
response.put("port", port);
|
||
response.put("msg", "评测完成");
|
||
|
||
logger.debug("直接执行任务!task: {}, tpi: {}", task, tpiID);
|
||
} else {
|
||
// 否则,将构建任务推入redis
|
||
JedisUtil.zpush("task", task);
|
||
|
||
//评测任务等待执行
|
||
response.put("ableToCreate", 0);
|
||
//在等待队列中的位置
|
||
response.put("waitNum", JedisUtil.zlen("task") + maxRunningPodNum);
|
||
response.put("code", 0);
|
||
response.put("msg", "等待评测");
|
||
|
||
logger.debug("任务入队等待!task: {}, tpi: {}", task, tpiID);
|
||
}
|
||
|
||
return response;
|
||
}
|
||
|
||
|
||
/**
|
||
* tpm版本库已更新,同步
|
||
*
|
||
* @param tpiID
|
||
* @param tpiGitURL
|
||
* @param tpmGitURL
|
||
* @return
|
||
*/
|
||
@RequestMapping(path = "/resetTpmRepository", method = RequestMethod.POST)
|
||
@ApiOperation(value = "tpm版本库已更新,同步操作", httpMethod = "POST", produces = MediaType.APPLICATION_FORM_URLENCODED_VALUE)
|
||
public JSONObject reset(
|
||
@ApiParam(name = "tpiID", required = true, value = "tpi") @RequestParam String tpiID,
|
||
@ApiParam(name = "tpiGitURL", required = true, value = "学员对应当前实训的版本库地址,base64编码") @RequestParam String tpiGitURL,
|
||
@ApiParam(name = "tpmGitURL", required = true, value = "学员对应当前实训的tpm版本库地址,base64编码") @RequestParam String tpmGitURL,
|
||
@ApiParam(name = "identifier", required = true, value = "push权限") @RequestParam String identifier)
|
||
throws Exception {
|
||
|
||
logger.debug("tpm版本库已更新,同步tpi版本库,tpiID:{}", tpiID);
|
||
|
||
JSONObject response = new JSONObject();
|
||
|
||
tpmGitURL = Base64Util.decode(tpmGitURL);
|
||
tpiGitURL = Base64Util.decode(tpiGitURL);
|
||
String tpiRepoName = StringHelper.getRepoName(tpiGitURL);
|
||
String path = appConfig.getWorkspace() + File.separator + "myshixun_" + tpiID;
|
||
|
||
try {
|
||
// 从tpm远程库拉取代码到myshixun版本库然后强推到tpi远程库
|
||
gameService.gitPullFromTpm(path, tpmGitURL, tpiRepoName);
|
||
gameService.gitPushToTpi(path, tpiGitURL, identifier);
|
||
|
||
logger.debug("tpm库更新内容已同步,tpiID:{}", tpiID);
|
||
response.put("code", 0);
|
||
response.put("msg", "版本库更新成功");
|
||
} catch (GameException e) {
|
||
logger.debug("tpm库更新内容同步失败, tpiID:{}", tpiID);
|
||
response.put("code", -1);
|
||
response.put("msg", "版本库同步失败!");
|
||
}
|
||
|
||
return response;
|
||
}
|
||
|
||
/**
|
||
* tpi版本库head是否存在,若缺失则修复
|
||
*
|
||
* @param tpiID
|
||
* @param tpiGitURL
|
||
* @return
|
||
* @throws Exception
|
||
*/
|
||
@RequestMapping(path = "/check")
|
||
@ApiOperation(value = "进入实训时做校验", httpMethod = "POST", produces = MediaType.APPLICATION_FORM_URLENCODED_VALUE)
|
||
public JSONObject check(
|
||
@ApiParam(name = "tpiID", required = true, value = "实训实例的ID") @RequestParam String tpiID,
|
||
@ApiParam(name = "tpiGitURL", required = true, value = "学员对应当前实训的版本库地址,base64编码") @RequestParam String tpiGitURL)
|
||
throws Exception {
|
||
|
||
JSONObject response = new JSONObject();
|
||
|
||
tpiGitURL = Base64Util.decode(tpiGitURL);
|
||
String tpiRepoName = StringHelper.getRepoName(tpiGitURL);
|
||
String tpiRepoPath = appConfig.getWorkspace() + "/myshixun_" + tpiID + "/" + tpiRepoName;
|
||
|
||
// 远程tpi库是否缺失head了
|
||
boolean noHead = ShellUtil.execute("cd " + tpiRepoPath + " && git remote show origin").contains("unknown");
|
||
if (noHead) {
|
||
logger.debug("tpiID:{} 远程tpi版本库head丢失", tpiID);
|
||
|
||
String identifier = StringHelper.getIdentifier(tpiGitURL);
|
||
|
||
// 处理仓库head丢失 ssh -p1122 git@10.9.191.219 'cd /home/git/repositories/p79061248/klp26sqc.git/refs/tmp/;
|
||
// cd `ls -t | sed -n "2p"`; cat head >../../heads/master; git update-ref HEAD `cat head`'
|
||
String command = "ssh -p1122 git@" + appConfig.getGitIP() + " 'cd repositories/" + identifier + "/"
|
||
+ tpiRepoName + ".git/refs/tmp; cd `ls -t | sed -n \"2p\"`; cat head >../../heads/master; git update-ref HEAD `cat head`'";
|
||
JSONObject result = ShellUtil.executeAndGetExitStatus(command);
|
||
|
||
response.put("fixHead", result.getIntValue("exitStatus"));
|
||
}
|
||
|
||
response.put("msg", "finished");
|
||
return response;
|
||
}
|
||
|
||
}
|