gradle 构建项目添加版本信息
gradle 构建项目添加版本信息,打包使用 spring boot 的打包插件
build.gradle 配置文件
bootJar {manifest {attributes('Project-Name': project.name,'Project-Version': project.version,"project-Vendor": "XXX Corp","Built-By": "Gradle ${gradle.gradleVersion}","Built-At": new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date()),'Git-Branch': getGitBranch(),'Git-Commit': getGitCommit(),'Git-Commit-Time': getGitCommitTime())}
}// 获取当前分支名称
static def getGitBranch() {try {def branch = 'git rev-parse --abbrev-ref HEAD'.execute().text.trim()return branch ?: 'unknown'} catch (Exception e) {System.err.println("fail to get branch, errMsg:" + e.getMessage())return 'unknown'}
}// 获取当前提交的记录的commitId
static def getGitCommit() {try {def commit = 'git rev-parse HEAD'.execute().text.trim()return commit ?: 'unknown'} catch (Exception e) {System.err.println("fail to get commit id, errMsg:" + e.getMessage())return 'unknown'}
}// 获取当前Git提交的时间
static def getGitCommitTime() {try {def commitTime = 'git show -s --format=%ct HEAD'.execute().text.trim()return commitTime ? new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date(Long.parseLong(commitTime) * 1000L)) : 'unknown'} catch (Exception e) {System.err.println("fail to get commit time, errMsg:" + e.getMessage())return 'unknown'}
}
版本信息模型
public class VersionModel {private String projectName;private String projectVersion;private String projectVendor;private String builtBy;private String builtAt;private String gitBranch;private String gitCommit;private String gitCommitTime;public String getProjectName() {return projectName;}public void setProjectName(String projectName) {this.projectName = projectName;}public String getProjectVersion() {return projectVersion;}public void setProjectVersion(String projectVersion) {this.projectVersion = projectVersion;}public String getProjectVendor() {return projectVendor;}public void setProjectVendor(String projectVendor) {this.projectVendor = projectVendor;}public String getBuiltBy() {return builtBy;}public void setBuiltBy(String builtBy) {this.builtBy = builtBy;}public String getBuiltAt() {return builtAt;}public void setBuiltAt(String builtAt) {this.builtAt = builtAt;}public String getGitBranch() {return gitBranch;}public void setGitBranch(String gitBranch) {this.gitBranch = gitBranch;}public String getGitCommit() {return gitCommit;}public void setGitCommit(String gitCommit) {this.gitCommit = gitCommit;}public String getGitCommitTime() {return gitCommitTime;}public void setGitCommitTime(String gitCommitTime) {this.gitCommitTime = gitCommitTime;}@Overridepublic String toString() {return "VersionModel{" +"projectName='" + projectName + '\'' +", projectVersion='" + projectVersion + '\'' +", implementationVendor='" + projectVendor + '\'' +", builtBy='" + builtBy + '\'' +", builtAt='" + builtAt + '\'' +", gitBranch='" + gitBranch + '\'' +", gitCommit='" + gitCommit + '\'' +", gitCommitTime='" + gitCommitTime + '\'' +'}';}
}
版本工具类
import com.alibaba.fastjson.JSONObject;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;import java.io.IOException;
import java.util.jar.Attributes;
import java.util.jar.Manifest;public class BuildUtils {public static final String projectName = "Project-Name";public static final String projectVersion = "Project-Version";public static final String projectVendor = "Project-Vendor";public static final String builtBy = "Built-By";public static final String builtAt = "Built-At";public static final String gitBranch = "Git-Branch";public static final String gitCommit = "Git-Commit";public static final String gitCommitTime = "Git-Commit-Time";private static final Logger log = LoggerFactory.getLogger(BuildUtils.class);public static String getManifestAttribute(String attributeName) {try {Manifest manifest = new Manifest(BuildUtils.class.getResourceAsStream("/META-INF/MANIFEST.MF"));Attributes attributes = manifest.getMainAttributes();return attributes.getValue(attributeName);} catch (IOException e) {log.warn("fail to get info from manifest file, errMsg:{}", e.getMessage(), e);return "unknown";}}public static Attributes getManifestAttribute() {try {Manifest manifest = new Manifest(BuildUtils.class.getResourceAsStream("/META-INF/MANIFEST.MF"));return manifest.getMainAttributes();} catch (IOException e) {log.warn("fail to get info from manifest file, errMsg:{}", e.getMessage(), e);return new Attributes();}}public static String getManifestAttributeJsonFormat() {return JSONObject.toJSONString(getVersionModel(), true);}public static VersionModel getVersionModel() {VersionModel versionModel = new VersionModel();try {Manifest manifest = new Manifest(BuildUtils.class.getResourceAsStream("/META-INF/MANIFEST.MF"));versionModel.setBuiltAt(manifest.getMainAttributes().getValue(builtAt));versionModel.setBuiltBy(manifest.getMainAttributes().getValue(builtBy));versionModel.setProjectName(manifest.getMainAttributes().getValue(projectName));versionModel.setProjectVersion(manifest.getMainAttributes().getValue(projectVersion));versionModel.setProjectVendor(manifest.getMainAttributes().getValue(projectVendor));versionModel.setGitBranch(manifest.getMainAttributes().getValue(gitBranch));versionModel.setGitCommit(manifest.getMainAttributes().getValue(gitCommit));versionModel.setGitCommitTime(manifest.getMainAttributes().getValue(gitCommitTime));} catch (IOException e) {log.warn("fail to get info from manifest file, errMsg:{}", e.getMessage(), e);}return versionModel;}
}
方法1 controller 示例
@RestController
@RequestMapping("/version")
public class VersionController {@GetMapping("/build")public VersionModel getVersionModel(){return BuildUtils.getVersionModel();}
}
方法2 执行 jar 包中的类执行获取版本信息
package com.version;
public class Version {public static void main(String[] args) {System.out.println(BuildUtils.getManifestAttributeJsonFormat());}
}
java -cp you-app.jar -Dloader.main=com.version.Version org.springframework.boot.loader.PropertiesLauncher
参考连接
https://blog.csdn.net/russle/article/details/130658805
相关文章:
gradle 构建项目添加版本信息
gradle 构建项目添加版本信息,打包使用 spring boot 的打包插件 build.gradle 配置文件 bootJar {manifest {attributes(Project-Name: project.name,Project-Version: project.version,"project-Vendor": "XXX Corp","Built-By": &…...
vue3 学习笔记17 -- 基于el-menu封装菜单
vue3 学习笔记17 – 基于el-menu封装菜单 前提条件:组件创建完成 配置路由 // src/router/index.ts import { createRouter, createWebHashHistory } from vue-router import type { RouteRecordRaw } from vue-router export const Layout () > import(/lay…...
使用 Redis 实现验证码、token 的存储,用自定义拦截器完成用户认证、并使用双重拦截器解决 token 刷新的问题
可以看一下我以前做过的笔记:黑马点评 短信登录部分 基于session实现登录流程 1.发送验证码 用户在提交手机号后,会校验手机号是否合法,如果不合法,则要求用户重新输入手机号 如果手机号合法,后台此时生成对应的验…...
反转链表 - 力扣(LeetCode)C语言
206. 反转链表 - 力扣(LeetCode)( 点击前面链接即可查看题目) /*** Definition for singly-linked list.* struct ListNode {* int val;* struct ListNode *next;* };*/ struct ListNode* reverseList(struct ListNode* head) {if(head NULL)…...
【Linux】进程间通信(1):进程通信概念与匿名管道
人与人之间是如何通信的?举个简单的例子,假如我是月老,我要为素不相识的但又渴望爱情的男女两方牵红线。我需要收集男方的信息告诉女方,收集女方的信息告诉男方,然后由男女双方来决定是否继续。对于他们而言࿰…...
Spring从入门到精通 01
文章目录 1. 依赖注入 (Dependency Injection, DI)2. 面向切面编程 (Aspect-Oriented Programming, AOP)3. 事务管理4. 简化 JDBC 开发5. 集成各种框架和技术6. 模块化和扩展性:主要的 Spring 模块:Core Container:AOP 模块:Data …...
C语言经典习题25
冒泡排序 对一维数组进行升序排序,然后在数组中输入20个数,将排序后的结果打印输出。 #include<stdio.h> #define N 20 int main() {int a[N];int i;for(i0;i<N;i) //初始化数组的数 {scanf("%d",&a);}for(i0;…...
2-47 基于matlab的时域有限差分法(FDTD法)拉夫等效原理进行时谐场外推
基于matlab的时域有限差分法(FDTD法)拉夫等效原理进行时谐场外推。外推边界距离吸收边界的距离、电磁场循环、傅立叶变换提起幅值和相位、各远区剖分点电场、方向系数计算等操作,得出可视化结果。程序已调通,可直接运行。 2-47 时域有限差分法(FDTD法) 拉…...
JupyterNotebook快捷键 自用
COMMAND MODE —————————————————————————————— Up Down cells的上下选择 A B 在上/下方插入cell C V X 复制/粘贴/剪切cell 双击D 删除所选cell Z 恢复被删除的cell 双击I Interrupt中断内核 Shift Enter 运行cell并选择下方 EDIT MODE ———…...
【我的OpenGL学习进阶之旅】讲一讲GL_TEXTURE_2D和GL_TEXTURE_EXTERNAL_OES的区别
在使用OpenGL ES进行图形图像开发时,我们常使用GL_TEXTURE_2D纹理类型,它提供了对标准2D图像的处理能力。这种纹理类型适用于大多数场景,可以用于展示静态贴图、渲染2D图形和进行图像处理等操作。 另外,有时我们需要从Camera或外部视频源读取数据帧并进行处理。这时,我们…...
Makefile 如何将生成的 .o 文件放到指定文件夹
研究了不少文章,我行通了一个,但是也不全,目前只能适用当前文件夹,如果源文件有子文件夹处理不了,还得继续研究。很多人说编译完把O文件移动走或者直接删掉。我想说的是不符合我的要求,移走或者删除O文件&a…...
聊一聊知识图谱结合RAG
因为最近在做一些关于提高公司内部使用的聊天机器人的回答准确率,并且最近微软官方也是开源了一下graphrag的源码,所以想聊一聊这个知识图谱结合rag。 rag在利用私有数据增强大模型回答的领域是一种比较典型的技术,也就是我们提出问题的时候&…...
Java面试锦集 之 一、Java基础(1)
一、Java基础(1) 1.final 关键字的作用? 修饰变量: 一旦被赋值,就不能再被修改,保证了变量值的稳定性。 例: final int NUMBER 10; //之后就不能再改变 NUMBER 的值了。修饰方法:…...
【leetcode】排列序列
给出集合 [1,2,3,...,n],其所有元素共有 n! 种排列。 按大小顺序列出所有排列情况,并一一标记,当 n 3 时, 所有排列如下: "123""132""213""231""312""321" 给定…...
【Cesium开发实战】视频融合功能的实现,可自定义位置和视频路径
Cesium有很多很强大的功能,可以在地球上实现很多炫酷的3D效果。今天给大家分享一个视频融合功能。 1.话不多说,先展示 视频融合 2.设计思路 点击绘制开始在地图上绘制视频融合的点位,形成视频播放的区域,双击弹框输入名称和要播放视频的路径,即可对应区域播放对应视频,…...
【秋招笔试题】小明的美食
解析:思维题。由于需要互不相同,每次操作取重复的值与最大值相加即可,这样即可保证相加后不会新增重复的值。因此统计重复值即可。 #include <iostream> #include <algorithm>using namespace std; const int maxn 1e5 5; int…...
基于OpenLCA、GREET、R语言的生命周期评价方法、模型构建及典型案例应用
生命周期分析 (Life Cycle Analysis, LCA) 是评价一个产品系统生命周期整个阶段——从原材料的提取和加工,到产品生产、包装、市场营销、使用、再使用和产品维护,直至再循环和最终废物处置——的环境影响的工具。这种方法被认为是一种“从摇篮到坟墓”的…...
Linux操作系统 -socket网络通信
同一台主机之间的进程 1.古老的通信方式 无名管道 有名管道 信号 2、IPC对象通信 system v 消息队列 共享内存 信号量集 由于不同主机间进程通信 3.socket网络通信 国际网络体系结构: 七层OSI模型(理论…...
【苍穹】完美解决由于nginx更换端口号导致无法使用Websocket
一、报错信息 进行到websocket开发的过程中,遇到了前端报错,无法连接的提示: 经过F12排查很明显是服务端和客户端并没有连接成功。这里就涉及到之前的坑,现在需要填上了。 二、报错原因和推导 应该还记得刚开苍穹的第一天配置前…...
Qt中在pro中实现一些宏定义
在pro文件中利用 DEFINES 定义一些宏定义供工程整体使用。(和在cpp/h文件文件中定义使用有点类似)可以利用pro的中的宏定义实现一些全局的判断 pro中实现 #自定义一个变量 DEFINES "PI\"3.1415926\"" #自定义宏 DEFINES "T…...
蓝桥杯嵌入式STM32G431RBT6保姆级外设配置指南(HAL库版,含LCD、ADC、PWM)
蓝桥杯嵌入式STM32G431RBT6实战开发全攻略:从HAL库配置到竞赛项目搭建 第一次拿到DK117E-G4开发板时,面对密密麻麻的引脚和陌生的HAL库,大多数嵌入式初学者都会感到无从下手。本文将带你从零开始,用最直观的方式掌握STM32G431RBT6…...
Brush 3D 重建引擎:多系统兼容、功能强大,渲染训练速度比 gsplat 更快!
特性训练方面,Brush 可接受 COLMAP 数据或 Nerfstudio 格式的数据集,在本地、移动端和浏览器中都能完全支持训练。训练时可与场景交互,实时查看训练动态,对比渲染效果与输入视图,还支持对带透明度的图像进行遮罩处理。…...
别再被EC11编码器波形坑了!STM32F103外部中断驱动避坑指南(附完整代码)
EC11编码器驱动开发实战:从硬件滤波到软件防抖的全方位避坑指南 旋转编码器作为人机交互的重要组件,在嵌入式系统中应用广泛。EC11以其性价比和可靠性成为许多项目的首选,但实际开发中,工程师常被信号抖动、方向误判等问题困扰。本…...
如何高效获取金融数据:Python通达信接口的完整指南
如何高效获取金融数据:Python通达信接口的完整指南 【免费下载链接】mootdx 通达信数据读取的一个简便使用封装 项目地址: https://gitcode.com/GitHub_Trending/mo/mootdx 在金融数据分析和量化交易领域,获取准确、及时且成本可控的市场数据一直…...
LinkSwift网盘直链下载助手:八大平台高速下载解决方案
LinkSwift网盘直链下载助手:八大平台高速下载解决方案 【免费下载链接】Online-disk-direct-link-download-assistant 一个基于 JavaScript 的网盘文件下载地址获取工具。基于【网盘直链下载助手】修改 ,支持 百度网盘 / 阿里云盘 / 中国移动云盘 / 天翼…...
UE4 TCP通信实战:从网络调试助手到Python服务端的跨平台数据交互
1. UE4 TCP通信基础与环境搭建 第一次在UE4里折腾TCP通信的时候,我对着文档研究了整整两天。后来发现其实用对方法,半小时就能跑通第一个Demo。这里分享我最常用的TCPSocketPlugin插件方案,比原生C实现简单十倍。 先到虚幻商城搜索"TCP …...
Elsevier Tracker:科研工作者必备的智能投稿状态追踪工具
Elsevier Tracker:科研工作者必备的智能投稿状态追踪工具 【免费下载链接】Elsevier-Tracker 项目地址: https://gitcode.com/gh_mirrors/el/Elsevier-Tracker 作为科研工作者,您是否曾因频繁登录Elsevier投稿系统查看审稿进度而感到疲惫&#x…...
开源项目发布自动化:GitHub与ClawHub技能包一键发布工具详解
1. 项目概述与核心价值如果你和我一样,经常需要将本地开发的项目,尤其是那些为ClawHub平台准备的技能包,发布到GitHub并同步推送到ClawHub技能市场,那你一定对下面这个场景不陌生:每次发布前,都要在脑子里重…...
开源首发:DocCenter — 本地 HTML 工作台,治好 AI 时代的文档散落病
Tags:Python aiohttp 开源项目 AI工具 前端工程 全栈 工具分享 一、痛点:AI 时代的文档散落病 (对比传统文档管理 vs AI 生成文档的区别,说明为什么 VSCode/Notion 都不合适) 二、技术选型:为什么是单 Pyth…...
ARM架构SUB与SUBS减法指令详解
1. ARM减法指令概述在ARM架构中,SUB(减法)和SUBS(带标志位的减法)是最基础的算术指令之一。这些指令用于执行寄存器值的减法操作,是构建更复杂算术运算和控制流程的基础。SUB指令从目标寄存器中减去一个立即…...
