将本地jar 批量发布到 Nexus 私服
ninehua 2024-12-13 15:33 95 浏览
前言
在日常开发过程中,我们有遇到将项目依赖通过 mvn deploy 发布到自己的公司的私服,以便其他人/环境统一下载入口。但是也有一些特殊情况,比如只能拿到第三方(或甲方)的依赖jar ,并不能拿到源码,所以只能将jar 通过 mvn deploy:deploy-file 发布到nexus,供构建环境下载依赖,一个两个还好,那几十个呢?所以,自己就用go 写了个发布工具。
问题/需求描述
项目性质决定我们的开发模式(苦逼项目外包),由甲方提供基础框架,我们基于基础框架进行业务开发,但是甲方不提供源码或者编译构建环境,需要申请VPN 已经访问账号,如果对方的VPN 网络不通畅很容易影响开发进度;而且,我们自己的构建环境没法通过他们的VPN 去下载依赖,所以,需要解决以下问题:
- 连甲方的VPN 下载基础框架的依赖(十几个包,而且还是 SNAPSHOT)
- 将基础框架的依赖发布到我们私服
解决方案
使用 mvn deploy:deploy-file命令发布到私服,我们先看看需要哪些参数:
- -Dfile需要发布jar包的绝对路径
- -DgroupId 对应pom 里面的 <groupId>groupId</groupId>
- -DartifactId 对应pom 里面的 <artifactId>artifactId</artifactId>
- -Dversion 对应pom 里面的 <version>version</version>
- -Dpackaging 对应pom 里面的 <packaging>packaging</packaging> ,比如 jar、pom
- -DrepositoryId 仓库ID
- -Durl 远程仓库地址
经过观察,我们本地仓库的依赖目录结构,会有两个关键内容,一个是jar 编译包,另个就是当前jar的maven 描述,如下图:
jar 都知道是需要发布到私服的文件,最关键的是 xxx-xxx-xxx.pom 描述文件,我们可以通过解析这个描述文件拿到 maven的 groupId、artifactId、version关键信息。
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<!-- This module was also published with a richer model, Gradle metadata, -->
<!-- which should be used instead. Do not delete the following line which -->
<!-- is to indicate to Gradle or any Gradle module metadata file consumer -->
<!-- that they should prefer consuming it instead. -->
<!-- do_not_remove: published-with-gradle-metadata -->
<modelVersion>4.0.0</modelVersion>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot</artifactId>
<version>2.7.6</version>
<name>spring-boot</name>
<description>Spring Boot</description>
<url>https://spring.io/projects/spring-boot</url>
<organization>
<name>Pivotal Software, Inc.</name>
<url>https://spring.io</url>
</organization>
<licenses>
<license>
<name>Apache License, Version 2.0</name>
<url>https://www.apache.org/licenses/LICENSE-2.0</url>
</license>
</licenses>
<developers>
<developer>
<name>Pivotal</name>
<email>info@pivotal.io</email>
<organization>Pivotal Software, Inc.</organization>
<organizationUrl>https://www.spring.io</organizationUrl>
</developer>
</developers>
<scm>
<connection>scm:git:git://github.com/spring-projects/spring-boot.git</connection>
<developerConnection>scm:git:ssh://git@github.com/spring-projects/spring-boot.git</developerConnection>
<url>https://github.com/spring-projects/spring-boot</url>
</scm>
<issueManagement>
<system>GitHub</system>
<url>https://github.com/spring-projects/spring-boot/issues</url>
</issueManagement>
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-core</artifactId>
<version>5.3.24</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.3.24</version>
<scope>compile</scope>
</dependency>
</dependencies>
</project>
分析完成之后,就可以知道代码的实现逻辑了,读取某个目录下的所有文件(含子目录),文件包含jar 以及 pom.xml 文件获取maven 坐标以及版本信息,代码:
// 查找需要发布的jar文件
func findDeployFile() ([]DeployFile, error) {
var deployeFilesSlice []DeployFile
err := filepath.Walk(RootPath, func(path string, info fs.FileInfo, err error) error {
if err != nil {
log.Fatal(err)
return err
}
if !info.IsDir() {
s := strings.Split(path, "\\")
// 取上一层目录,作为版本
version := s[len(s)-2 : len(s)-1]
if strings.Contains(info.Name(), version[0]) && strings.HasSuffix(info.Name(), ".pom") {
p, _ := getPom(path)
if strings.Contains(info.Name(), p.Version) {
var _path = path
if p.Packaging == "jar" {
_path = strings.Replace(_path, ".pom", ".jar", 1)
}
var depolyeFile = DeployFile{FilePath: _path, PomConfig: p}
deployeFilesSlice = append(deployeFilesSlice, depolyeFile)
}
}
}
return nil
})
if err != nil {
return nil, err
}
return deployeFilesSlice, nil
}
// 读取pom 文件,获取坐标
func getPom(path string) (Pom, error) {
fmt.Printf("path: %v\n", path)
file, err := os.Open(path)
if err != nil {
log.Fatal(err)
return Pom{}, err
}
// 关闭流
defer file.Close()
// 读取xml 文件
b, err2 := ioutil.ReadAll(file)
if err2 != nil {
log.Fatal(err2)
return Pom{}, err2
}
s := strings.Split(path, "\\")
v := Pom{}
err3 := xml.Unmarshal(b, &v)
if err != nil {
log.Fatal(err3)
}
if v.GroupId == "" {
v.GroupId = v.Parent.GroupId
}
if v.Packaging == "" {
v.Packaging = "jar"
}
if v.Version == "" {
// 取上一层目录,作为版本
version := s[len(s)-2 : len(s)-1]
v.Version = version[0]
}
fmt.Printf("version: %v\n", v.Version)
// fmt.Printf("v: %v\n", v)
return v, nil
}
拿到需要发布jar 的坐标关键信息之后,就是拼接 mvn deploy:deploy-file的命令并执行该命令。代码:
// 发布到私服
func deployeCMD(deployConfig DeployFile) {
var sourceFilePath = ""
if deployConfig.PomConfig.Packaging == "jar" {
sourceFilePath = "-Dsources=" + strings.Replace(deployConfig.FilePath, ".jar", "-sources.jar", 1)
}
cmd := exec.Command("mvn",
"deploy:deploy-file",
"-Dmaven.test.skip=true",
"-Dfile="+deployConfig.FilePath,
sourceFilePath,
"-DgroupId="+deployConfig.PomConfig.GroupId,
"-DartifactId="+deployConfig.PomConfig.ArtifactId,
"-Dversion="+deployConfig.PomConfig.Version,
"-Dpackaging="+deployConfig.PomConfig.Packaging,
"-DrepositoryId="+RerepositoryId,
"-Durl="+RemoteURL)
fmt.Printf("cmd.Args: %v\n", cmd.Args)
stdout, err := cmd.StdoutPipe()
if err != nil {
log.Fatal(err)
}
// 关闭流
defer stdout.Close()
// 输出命令执行结果
if err := cmd.Start(); err != nil { // 运行命令
log.Fatal(err)
}
if opBytes, err := ioutil.ReadAll(stdout); err != nil { // 读取输出结果
log.Fatal(err)
return
} else {
log.Println(string(opBytes))
}
}
到这里,将本地jar批量发布到nexus 的代码就算敲完了,下面附上完整的代码。
package main
import (
"encoding/xml"
"fmt"
"io/fs"
"io/ioutil"
"log"
"os"
"os/exec"
"path/filepath"
"strings"
)
type Pom struct {
XMLName xml.Name `xml:"project"`
GroupId string `xml:"groupId"`
ArtifactId string `xml:"artifactId"`
Version string `xml:"version"`
Packaging string `xml:"packaging"`
Parent Parent `xml:"parent"`
}
type Parent struct {
GroupId string `xml:"groupId"`
ArtifactId string `xml:"artifactId"`
Version string `xml:"version"`
}
type DeployFile struct {
FilePath string
PomConfig Pom
}
const (
RootPath string = "D:\\Baas\\leatop_jar\\leatop"
RemoteURL string = "https://middle.xxxx.cn/repository/mvn-leatop-snapshots/"
RerepositoryId string = "mvn-leatop-snapshots"
)
// 读取pom 文件,获取坐标
func getPom(path string) (Pom, error) {
fmt.Printf("path: %v\n", path)
file, err := os.Open(path)
if err != nil {
log.Fatal(err)
return Pom{}, err
}
// 关闭流
defer file.Close()
// 读取xml 文件
b, err2 := ioutil.ReadAll(file)
if err2 != nil {
log.Fatal(err2)
return Pom{}, err2
}
s := strings.Split(path, "\\")
v := Pom{}
err3 := xml.Unmarshal(b, &v)
if err != nil {
log.Fatal(err3)
}
if v.GroupId == "" {
v.GroupId = v.Parent.GroupId
}
if v.Packaging == "" {
v.Packaging = "jar"
}
if v.Version == "" {
// 取上一层目录,作为版本
version := s[len(s)-2 : len(s)-1]
v.Version = version[0]
}
fmt.Printf("version: %v\n", v.Version)
// fmt.Printf("v: %v\n", v)
return v, nil
}
// 查找需要发布的jar文件
func findDeployFile() ([]DeployFile, error) {
var deployeFilesSlice []DeployFile
err := filepath.Walk(RootPath, func(path string, info fs.FileInfo, err error) error {
if err != nil {
log.Fatal(err)
return err
}
if !info.IsDir() {
s := strings.Split(path, "\\")
// 取上一层目录,作为版本
version := s[len(s)-2 : len(s)-1]
if strings.Contains(info.Name(), version[0]) && strings.HasSuffix(info.Name(), ".pom") {
p, _ := getPom(path)
if strings.Contains(info.Name(), p.Version) {
var _path = path
if p.Packaging == "jar" {
_path = strings.Replace(_path, ".pom", ".jar", 1)
}
var depolyeFile = DeployFile{FilePath: _path, PomConfig: p}
deployeFilesSlice = append(deployeFilesSlice, depolyeFile)
}
}
}
return nil
})
if err != nil {
return nil, err
}
return deployeFilesSlice, nil
}
// 发布到私服
func deployeCMD(deployConfig DeployFile) {
var sourceFilePath = ""
if deployConfig.PomConfig.Packaging == "jar" {
sourceFilePath = "-Dsources=" + strings.Replace(deployConfig.FilePath, ".jar", "-sources.jar", 1)
}
cmd := exec.Command("mvn",
"deploy:deploy-file",
"-Dmaven.test.skip=true",
"-Dfile="+deployConfig.FilePath,
sourceFilePath,
"-DgroupId="+deployConfig.PomConfig.GroupId,
"-DartifactId="+deployConfig.PomConfig.ArtifactId,
"-Dversion="+deployConfig.PomConfig.Version,
"-Dpackaging="+deployConfig.PomConfig.Packaging,
"-DrepositoryId="+RerepositoryId,
"-Durl="+RemoteURL)
fmt.Printf("cmd.Args: %v\n", cmd.Args)
stdout, err := cmd.StdoutPipe()
if err != nil {
log.Fatal(err)
}
// 关闭流
defer stdout.Close()
// 输出命令执行结果
if err := cmd.Start(); err != nil { // 运行命令
log.Fatal(err)
}
if opBytes, err := ioutil.ReadAll(stdout); err != nil { // 读取输出结果
log.Fatal(err)
return
} else {
log.Println(string(opBytes))
}
}
func main() {
df, _ := findDeployFile()
for _, file := range df {
fmt.Printf("file: %v\n", file)
deployeCMD(file)
}
// cmd := exec.Command("mvn", "-version")
}
总结
在使用过程中需要注意的2点:
- windows 下与mac 的目录结构不一样,需要调整
- RootPath string = "D:\\Baas\\leatop_jar\\leatop" 这个目录不能直接是maven 本地仓库目录,需要单独出来
相关推荐
- 把Apple Watch/iPhone/Mac装进折叠屏:vivo X Fold5通讯体验
-
【ZOL中关村在线原创技术解析】在折叠屏手机硬件堆砌逐渐同质化的当下,vivoXFold5另辟蹊径,将通讯体验作为核心突破点——不仅实现安卓手机与苹果生态的跨系统通讯互联,更通过芯片级调校与创新技...
- CH9329双头线使用说明(双头线有什么用)
-
目录1.介绍说明2.测试说明3.修改为ASCII模式(CH9328字符模式)常见问题解答:1.介绍说明CH9329双头线是集成了CH9329+CH340芯片的成品线,主要作用是使用主控电脑发送串口指令...
- 电脑F1—F12功能键有什么作用?大多数人都不用?
-
对于经常运用电脑的人来说键盘肯定是最了如指掌的吧,那小编问你键盘上的F1-F12都有什么功能呢?是不是感觉好像知道但是却说不出来呢?今天小编就介绍一下F1—F12功能键的作用。F1帮助 当你在桌面...
- 笔记本键盘弹力测试机:提升产品质量的关键一环-磐石测控
-
在笔记本电脑的体验中,键盘手感往往被用户视为重要的评估指标之一。一个键程适中、弹力均衡、响应迅速的键盘不仅提升使用舒适度,还反映了产品的整体做工与质量。在键盘的设计与量产过程中,如何确保每一个按键都达...
- 无线键盘怎么连接电脑?简单指南,一文学会!
-
在现代办公和家庭环境中,无线键盘因其便利性和整洁的桌面布局而越来越受欢迎。无线键盘让桌面显得更加整洁,同时也提供了更大的操作灵活性。但你知道无线键盘怎么连接电脑吗?本文将为你提供一份简洁明了的指南,帮...
- 灯厂力作,电竞高手专属外设:雷蛇猎魂光蛛V3专业竞技版键盘实测
-
玩电脑,啥叫专业?你落地成盒,我毒圈吃鸡,我就比你专业。你录像整活,我赛事夺冠,我就比你专业。换言之,专业并不是一个绝对的定义,而是相对而言的、是分层级的。那么以此推理,我搞到一个专业的电竞外设,加上...
- 机械键盘新物种:Keyview13"Touch登场,13英寸触控屏 +SSD扩展
-
IT之家6月18日消息,科技媒体notebookcheck昨日(6月17日)发布博文,报道称美国品牌AuraDisplays推出创新机械键盘Keyview13"T...
- 键圈玩家的“退烧”好选择:燃风RC1 78键 + RT1数字小键盘评测
-
Hello,大家好!我是沈少!今天带来的是一套非常高品质的静电容键盘套装。这两款产品,分别是我目前的办公主力键盘:燃风RC1;以及快速输入数字的专用数字小键盘”RT1。当然了,作为两款独立的产品,小伙...
- US220 2口USB3.0切换器 两台电脑共用USB3.0连接设备连打印机共享器
-
登昌恒US2202口USB3.0切换器两台电脑共用USB3.0连接设备连打印机鼠标键盘共享器支持手动切换按键切换支持WINDOWSMAC...
- S7-1200以太网模块CP1243使用(西门子1200以太网通讯模块)
-
1概述S7-1200本体集成一个或者两个以太网端口,可以实现诸如ProfinetIO、S7、TCP、OPCUA等通信协议。此外还可以通过左边扩展插槽,通过添加CP1243-1模块实现以太网接口的...
- 公司内网打印机如何共享最便捷(公司内网连接打印机步骤)
-
我们在公司上班工作时经常会用到打印机打印文件,但是很多公司不会每个人配备一台打印机。为了充分提高资金利用率,都是一个办公室或几个人共用一台打印机。这个时候就需要如何共享一台打印机了?对于打印机共享,...
- 极空间新版虚拟机来袭,带来显卡效率大幅提升
-
极空间新版虚拟机来袭,带来显卡效率大幅提升哈喽小伙伴们好,我是Stark-C~极空间最近为了大家过年HAPPY可谓是操碎了心,不仅带来广受好评的极影视2.0,APP、网页、TV版的新播放器也是同步升...
- 电脑网络出现黄色感叹号?一文教你彻底解决无网络连接问题
-
当电脑右下角的网络图标突然亮起黄色感叹号,显示"无Internet连接"时,这种突如其来的断网状况往往让人措手不及。无论是正在进行的视频会议、即将提交的工作文件,还是在线游戏的关键时刻...
- 最另类802.11ac路由器:系统完爆所有!
-
Wi-Fi路由器正在朝智能化方向发展,但是NAS厂商群晖科技(Synology)的这一款“RT1900ac”着实是最为特别的,关键就在于系统与众不同。去年9月份的时候群晖就自办展示会,秀了一把这台路由...
- 电脑连不上网络怎么办?附一键修复的解决攻略
-
电脑连不上网络作为一种常见问题,虽然不难解决,但是电脑连不上网络的原因可能有很多,比如网线、网卡、路由器、调制解调器、网络设置等。要解决这个问题,我们需要根据不同的情况采取不同的方法。以下便是关于电脑...