以前作为后端,平时写好代码提交后就完事了,这周忽然叫我做公布到开发环境的操作,每次打包的复制到服务器上又很麻烦,尽管晓得Jenkins,然而这工具配置项也很多,就想着能不能自己做一个。于是就写起了代码。
想公布一个软件到服务器,先分步骤。
1.拉取代码
2.打包
3.复制到服务器
4.重启服务。
一开始,拉取代码的工作我是手动进行的,因为思考到要合并代码。这一步还是不要给程序做了,打包的话比较简单,间接
maven package
就好,顶多加一个
maven package -P dev(看你springboot里的配置)
打包完了必定要把包弄到服务器上,天然想到了scp命令,好在windows10是反对scp的,然而个别的scp须要明码,所以得把本人的公钥贴在服务器上。
登录到服务器
cd ~/.ssh
ls显示出
authorized_keys id_rsa id_rsa.pub known_hosts
把你windows的公钥追加到authorized_keys文件里就能够
于是我的第一个脚本是这样的
windows下新建bat
call "maven门路\mvn" package -P pro
这里新建了脚本是因为maven运行完窗口不会主动敞开,没法执行下一行,新启脚本的话能够解决这个问题
start /wait copy.bat
其中copy的bat是
scp C:\你的门路\你的我的项目.jar root@ip地址:/理论目录
看上去非常简单
把jar包拷贝过来了后怎么让他重启呢?
一开始想到用ssh近程执行命令,后果没找到不便又适合的计划,抉择了java解决。本人写了一个jar包当时丢上去。代码很简略
@Component
public class Task {
private String lastMd5 = "";
*//**
* 每隔20秒执行一次
*//*
@Scheduled(fixedRate = 20000)
public void testTasks() throws IOException {
String md5 = DigestUtils.md5DigestAsHex(new FileInputStream("你的服务器上包.jar"));
if(md5.equals(lastMd5)){
return;
}
System.out.println(md5);
lastMd5 = md5;
String[] cmd = new String[]{"/bin/sh", "-c", "ps -ef|grep 你的.jar |grep -v grep | awk '{print $2}' | xargs -r kill -9 && java -Dloader.path=/目录/lib -jar /你的服务器上包.jar"};
Process process = null;
try{
process = Runtime.getRuntime().exec(cmd);
}catch (Exception e){
}
}
}*/
首先阐明下,我maven打的是只蕴含class文件的包,不蕴含依赖的,所以一个包很小(不便传输和取md5)。
下面监听程序很简略,20秒跑一次,比照正在跑的jar包的md5(旧值)与传上去的jar包md5(新值),如果不一样就认为改变了,就调用bash脚本敞开原有的程序,而后重新启动程序。
到这里,我集体的主动部署曾经胜利了,每次拉取代码后想要更新,点一下bat,等几秒就行了。
共事看完感觉成果很不错,不过提了个倡议,你都上java了为什么不间接java把全副工作都做了,这语言咱们熟。
于是又进行了改版,找了下材料后,有了几个可用包。
1.JGIT(它的作用是拉取git代码)
<dependency>
<groupId>org.eclipse.jgit</groupId>
<artifactId>org.eclipse.jgit</artifactId>
<version>5.10.0.202012080955-r</version>
</dependency>
public class JgitUtil {
//验证
private static CredentialsProvider credentialsProvider;
//这个是我的项目拉取后的.git文件,用它来判断须要做git clone还是pull操作
private static String localGitUrl;
static {
credentialsProvider = new UsernamePasswordCredentialsProvider(PropUtils.get("username"), PropUtils.get("password"));
localGitUrl = System.getProperty("user.dir") + "/" + PropUtils.get("projectName") + "/.git";
}
/**
* git克隆
* @param gitUrl
* @param branch
* @throws GitAPIException
*/
public static void gitClone(String gitUrl, String branch) throws GitAPIException {
Git.cloneRepository()
.setURI(gitUrl)
.setBranch(branch)
.setCredentialsProvider(credentialsProvider)
.setDirectory(new File(System.getProperty("user.dir") + "/" + PropUtils.get("projectName")))
.call();
}
/**
* git拉取操作
* @throws IOException
* @throws GitAPIException
*/
public static void pull() throws IOException, GitAPIException {
//判断仓库是否曾经存在,不存在就clone
File local = new File(localGitUrl);
if(!local.exists()){
//第一个参数是git拉取的地址,第二个参数是分支名
gitClone(PropUtils.get("gitUrl"), PropUtils.get("branch"));
return;
}
//曾经存在就pull
Repository localRepo = new FileRepository(localGitUrl);
Git git = new Git(localRepo);
PullCommand pullCommand = git.pull();
pullCommand.setCredentialsProvider(credentialsProvider);
pullCommand.call();
}
}
2.maven-invoker(它的作用是打包我的项目)
<dependency>
<groupId>org.apache.maven.shared</groupId>
<artifactId>maven-invoker</artifactId>
<version>3.1.0</version>
</dependency>
private static InvocationRequest request;
private static Invoker invoker;
//须要设置maven目录
private static String mavenHome = System.getProperty("user.dir") + "/" + PropUtils.get("mavenName");
static {
request = new DefaultInvocationRequest();
//配置maven的settings.xml文件,外面仓库门路要配好
request.setUserSettingsFile(new File(mavenHome + "/conf/settings.xml"));
String env = PropUtils.get("env");
String packageStr = "package";
if(!StringUtils.isEmpty(env)){
packageStr += " -P " + env;
}
//这个设置maven命令,个别只用package
request.setGoals(Collections.singletonList(packageStr));
//设置我的项目的pom所在
File pomFile = new File(System.getProperty("user.dir") + "/" + PropUtils.get("projectName") + "/pom.xml");
if(!pomFile.exists()){
pomFile = new File(PropUtils.get("pomFile"));
}
request.setPomFile(pomFile);
//配置maven拉取的依赖所在的门路,我间接和maven放一起了
File repo = new File(mavenHome + "/repo");
if(!repo.exists()){
repo.mkdir();
}
request.setLocalRepositoryDirectory(repo);
invoker = new DefaultInvoker();
invoker.setMavenHome(new File(mavenHome));
}
//调用一下就能做拉取的操作
public static void mvnPackage() throws MavenInvocationException {
invoker.execute(request);
}
而后前端给个按钮,controller层就简略了
执行一下git拉取,执行一下maven打包
java用Runtime.getRuntime().exec(cmd)执行linux命令就能够了
linux命令包含杀死旧的java,命令在下面了, cp命令挪动打好的包,这里举荐用
/cp -ruv
后面带斜杠是要应用cp自身的命令,默认的cp命令有别名,理论调用的是cp -i,每次复制时会提醒要不要笼罩,加上斜杠就不会了。
复制好后重启jar包就能够了。
如果你想获取java调用linux命令后控制台的输入,能够这样
process = Runtime.getRuntime().exec(cmd);
LineNumberReader br = null;
br = new LineNumberReader(new InputStreamReader(
process.getInputStream()));
StringBuffer sb = new StringBuffer();
String line;
while ((line = br.readLine()) != null) {
sb.append(line).append("\n");
}
但不是很举荐,打包命令很多无用的输入,这个是我以前写网页端调用linux命令时查的,这里其实不太用得上,不过我本人还是把cp后果打印到前端了,次要是看下是否胜利了。
前端成果,就两个按钮,一个公布后端,一个公布前端(这个做好了,暂无两个字忘了删了)。嫌麻烦能够间接用接口申请,post man或者window间接用curl,再写个bat(笑)
点个[公布前端],能够看到把cp命令打印进去了。
发表回复