将本地jar 批量发布到 Nexus 私服

前言

在日常开发过程中,我们有遇到将项目依赖通过 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 描述,如下图:


image.png

jar 都知道是需要发布到私服的文件,最关键的是 xxx-xxx-xxx.pom 描述文件,我们可以通过解析这个描述文件拿到 maven的 groupIdartifactIdversion关键信息。

<?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 本地仓库目录,需要单独出来
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,277评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,689评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,624评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,356评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,402评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,292评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,135评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,992评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,429评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,636评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,785评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,492评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,092评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,723评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,858评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,891评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,713评论 2 354

推荐阅读更多精彩内容