spring boot添加License(软件许可)

文章目录

  • 前言
  • 1. 生成钥匙库
  • 2. 生成证书
  • 3. 生成公匙库
  • 4.业务代码
    • 1. 引入依赖
    • 2. 关键代码
    • 3. 配置文件
  • 5、改成线上地址,这样不用每次打包,发送license.lic文件给客户,重启项目就行
    • 5.1、工具类
    • 5.2 修改部分:
  • 总结


前言

工作需要给软件加上许可

keytool 工具在你的java安装的bin目录中,命令行打开执行就行

1. 生成钥匙库

# validity:私钥的有效期多少天
# alias:私钥别称
# keyalg:指定加密算法,默认是DSA
# keystore: 指定私钥库文件的名称(生成在当前目录)
# storepass:指定私钥库的密码(获取keystore信息所需的密码) 
# keypass:指定别名条目的密码(私钥的密码) keytool -genkeypair -keysize 1024 -validity 3650 -alias "privateKey" -keystore "C:\Users\Administrator\Desktop\license\privateKey.keystore" -storepass "zhiutech@123" -keypass "zhiutech@123" -dname "CN=localhost, OU=localhost, O=localhost, L=SH, ST=SH, C=CN"

2. 生成证书


# alias:私钥别称
# keystore:指定私钥库的名称(在当前目录查找)
# storepass: 指定私钥库的密码
# file:证书名称keytool -exportcert -alias "privateKey" -keystore "C:\Users\Administrator\Desktop\license\privateKey.keystore" -storepass "zhiutech@123" -file "C:\Users\Administrator\Desktop\license\certfile.cer"

3. 生成公匙库


# alias:公钥别称
# file:证书名称
# keystore:公钥文件名称
# storepass:指定私钥库的密码
keytool -import -alias "publicCert" -file "C:\Users\Administrator\Desktop\license\certfile.cer" -keystore "C:\Users\Administrator\Desktop\license\publicCerts.keystore" -storepass "zhiutech@123"

4.业务代码

源码借鉴了这个地址:

https://gitee.com/Zhiyun_Lee/ruo-yi-vue3-license

1. 引入依赖

<!-- License -->
<dependency><groupId>de.schlichtherle.truelicense</groupId><artifactId>truelicense-core</artifactId><version>1.33</version>
</dependency>

2. 关键代码

CustomKeyStoreParam自定义参数,公私钥存放路径和其他信息。关键在于重写getStream方法,会因为本地开发环境原因会报错,需要根据注释使用不同方法获取存储路径上的文件信息。

/*** 自定义KeyStoreParam*/
public class CustomKeyStoreParam extends AbstractKeyStoreParam {// ...省略代码/*** 重写getStream()方法* @return* @throws IOException*/@Overridepublic InputStream getStream() throws IOException {// 本地开发环境,License生成
//        final InputStream in = new FileInputStream(new File(storePath));// 本地开发环境,License加载// 线上环境,直接用这个InputStream in = new ClassPathResource(storePath).getStream();if (null == in){throw new FileNotFoundException(storePath);}return in;}
}


ResourcesConfig通用配置文件,在自定义拦截规则里添加License检查拦截器。考虑到License检查需要花费时间,如果所有接口都设置拦截,那意味着整个项目平均请求慢了几百毫秒,所以这边我只拦截登录接口,因为没有登录就没有token鉴权,也就没有可能请求成功其他接口。

/*** 通用配置* * @author ruoyi*/
@Configuration
public class ResourcesConfig implements WebMvcConfigurer
{@Autowiredprivate RepeatSubmitInterceptor repeatSubmitInterceptor;@Autowiredprivate LicenseCheckInterceptor licenseCheckInterceptor;// ...省略代码/*** 自定义拦截规则*/@Overridepublic void addInterceptors(InterceptorRegistry registry){registry.addInterceptor(repeatSubmitInterceptor).addPathPatterns("/**");registry.addInterceptor(licenseCheckInterceptor).addPathPatterns("/login");}// ...省略代码
}

SecurityConfig配置文件,添加/license/getInfo请求接口地址不过滤。

/*** spring security配置* * @author ruoyi*/
@EnableGlobalMethodSecurity(prePostEnabled = true, securedEnabled = true)
public class SecurityConfig extends WebSecurityConfigurerAdapter
{// ...省略代码@Overrideprotected void configure(HttpSecurity httpSecurity) throws Exception{httpSecurity// 对于登录login 注册register 验证码captchaImage 允许匿名访问// 添加获取License硬件信息.antMatchers("/login", "/register", "/captchaImage", "/license/getInfo").permitAll();}
}

3. 配置文件

application.yml配置文件填写License相关配置

#License相关配置
license:#License如果是本地地址,则改成本地地址,我这里改成线上地址了licenseFile: http://192.168.110.188:9000/iot/2024/05/21/license.licsubject: MyLicenseDemopublicAlias: publicCertstorePass: zhiutech@123# 公钥地址publicKeysStoreFile: C:/Users/Administrator/Desktop/license/publicCerts.keystoregenerateLicensePath: C:/Users/Administrator/Desktop/license/license.lic

提示:如果只是打包到文件中,到这个地方就可以了,下面的内容是改成线上访问授权文件,不是本地访问了

5、改成线上地址,这样不用每次打包,发送license.lic文件给客户,重启项目就行

这里我对licenseFile做了一下改变,如果您不需要线上地址去校验license.lic,到这个地方就可以了

5.1、工具类

package com.ruoyi.auth.license.utils;import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileItemFactory;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.poi.xwpf.usermodel.*;
import org.springframework.web.multipart.MultipartFile;
import org.springframework.web.multipart.commons.CommonsMultipartFile;import java.io.*;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;/*** 操作word文档工具类** @author wangyj* @date 2020-03-03**/
public class POIUtil {/*** 用一个docx文档作为模板,然后替换其中的内容,再写入目标文档中。* @throws Exception*/public static void templateWrite(String filePath,String outFilePath,Map<String, Object> params) throws Exception {InputStream is = new FileInputStream(filePath);XWPFDocument doc = new XWPFDocument(is);//替换段落里面的变量replaceInPara(doc, params);//替换表格里面的变量replaceInTable(doc, params);OutputStream os = new FileOutputStream(outFilePath);doc.write(os);close(os);close(is);}/*** 用一个docx文档作为模板,然后替换其中的内容,再写入目标文档中。* @throws Exception*/public static XWPFDocument templateWrite(InputStream is,Map<String, Object> params) throws Exception {XWPFDocument doc = new XWPFDocument(is);//替换段落里面的变量replaceInPara(doc, params);//替换表格里面的变量replaceInTable(doc, params);//保存文档close(is);return doc;}/*** XWPFDocument 转 MultipartFile(CommonsMultipartFile)** @param document 文档对象* @param fileName 文件名* @return*/public static MultipartFile xwpfDocumentToCommonsMultipartFile(XWPFDocument document, String fileName) {//XWPFDocument转FileItemFileItemFactory factory = new DiskFileItemFactory(16, null);FileItem fileItem = factory.createItem("textField", "application/vnd.openxmlformats-officedocument.wordprocessingml.document", true, fileName+".docx");try {OutputStream os = fileItem.getOutputStream();document.write(os);os.close();//FileItem转MultipartFileMultipartFile multipartFile = new CommonsMultipartFile(fileItem);return multipartFile;} catch (Exception e) {e.printStackTrace();}return null;}/*** 替换段落里面的变量* @param doc 要替换的文档* @param params 参数*/private static void replaceInPara(XWPFDocument doc, Map<String, Object> params) {Iterator<XWPFParagraph> iterator = doc.getParagraphsIterator();XWPFParagraph para;while (iterator.hasNext()) {para = iterator.next();replaceInPara(para, params);}}/*** 替换段落里面的变量*/public static List<String> getparas(InputStream is) throws IOException {XWPFDocument doc = new XWPFDocument(is);List<String> paramsList = new ArrayList<>();Iterator<XWPFParagraph> iterator = doc.getParagraphsIterator();XWPFParagraph para;while (iterator.hasNext()) {para = iterator.next();System.out.println(para.getParagraphText());Matcher matcher = matcher(para.getParagraphText());if (matcher.find()) {paramsList.add(matcher.group(1));}}return paramsList;}/*** 替换段落里面的变量** @param para   要替换的段落* @param params 参数*/private static void replaceInPara(XWPFParagraph para, Map<String, Object> params) {List<XWPFRun> runs;Matcher matcher;String runText = "";int fontSize = 0;UnderlinePatterns underlinePatterns = null;if (matcher(para.getParagraphText()).find()) {runs = para.getRuns();if (runs.size() > 0) {int j = runs.size();for (int i = 0; i < j; i++) {XWPFRun run = runs.get(0);if (fontSize == 0) {fontSize = run.getFontSize();}if(underlinePatterns==null){underlinePatterns=run.getUnderline();}String i1 = run.toString();runText += i1;para.removeRun(0);}}matcher = matcher(runText);if (matcher.find()) {while ((matcher = matcher(runText)).find()) {runText = matcher.replaceFirst(String.valueOf(params.get(matcher.group(1))));}//直接调用XWPFRun的setText()方法设置文本时,在底层会重新创建一个XWPFRun,把文本附加在当前文本后面,//所以我们不能直接设值,需要先删除当前run,然后再自己手动插入一个新的run。//para.insertNewRun(0).setText(runText);//新增的没有样式XWPFRun run = para.createRun();run.setText(runText,0);run.setFontSize(fontSize);run.setUnderline(underlinePatterns);run.setFontFamily("仿宋");//字体run.setFontSize(16);//字体大小//run.setBold(true); //加粗//run.setColor("FF0000");//默认:宋体(wps)/等线(office2016) 5号 两端对齐 单倍间距//run.setBold(false);//加粗//run.setCapitalized(false);//我也不知道这个属性做啥的//run.setCharacterSpacing(5);//这个属性报错//run.setColor("BED4F1");//设置颜色--十六进制//run.setDoubleStrikethrough(false);//双删除线//run.setEmbossed(false);//浮雕字体----效果和印记(悬浮阴影)类似//run.setFontFamily("宋体");//字体//run.setFontFamily("华文新魏", FontCharRange.cs);//字体,范围----效果不详//run.setFontSize(14);//字体大小//run.setImprinted(false);//印迹(悬浮阴影)---效果和浮雕类似//run.setItalic(false);//斜体(字体倾斜)//run.setKerning(1);//字距调整----这个好像没有效果//run.setShadow(true);//阴影---稍微有点效果(阴影不明显)//run.setSmallCaps(true);//小型股------效果不清楚//run.setStrike(true);//单删除线(废弃)//run.setStrikeThrough(false);//单删除线(新的替换Strike)//run.setSubscript(VerticalAlign.SUBSCRIPT);//下标(吧当前这个run变成下标)---枚举//run.setTextPosition(20);//设置两行之间的行间距//run.setUnderline(UnderlinePatterns.DASH_LONG);//各种类型的下划线(枚举)//run0.addBreak();//类似换行的操作(html的  br标签)//run0.addTab();//tab键//run0.addCarriageReturn();//回车键//注意:addTab()和addCarriageReturn() 对setText()的使用先后顺序有关:比如先执行addTab,再写Text这是对当前这个Text的Table,反之是对下一个run的Text的Tab效果}}}/*** 替换表格里面的变量* @param doc 要替换的文档* @param params 参数*/private static void replaceInTable(XWPFDocument doc, Map<String, Object> params) {Iterator<XWPFTable> iterator = doc.getTablesIterator();XWPFTable table;List<XWPFTableRow> rows;List<XWPFTableCell> cells;List<XWPFParagraph> paras;while (iterator.hasNext()) {table = iterator.next();rows = table.getRows();for (XWPFTableRow row : rows) {cells = row.getTableCells();for (XWPFTableCell cell : cells) {paras = cell.getParagraphs();for (XWPFParagraph para : paras) {replaceInPara(para, params);}}}}}/*** 正则匹配字符串* @param str* @return*/private static Matcher matcher(String str) {Pattern pattern = Pattern.compile("\\$\\{(.+?)\\}", Pattern.CASE_INSENSITIVE);Matcher matcher = pattern.matcher(str);return matcher;}/*** 关闭输入流* @param is*/private static void close(InputStream is) {if (is != null) {try {is.close();} catch (IOException e) {e.printStackTrace();}}}/*** 关闭输出流* @param os*/private static void close(OutputStream os) {if (os != null) {try {os.close();} catch (IOException e) {e.printStackTrace();}}}public static InputStream getImageStream(String url) {try {HttpURLConnection connection = (HttpURLConnection) new URL(url).openConnection();connection.setReadTimeout(5000);connection.setConnectTimeout(5000);connection.setRequestMethod("GET");if (connection.getResponseCode() == HttpURLConnection.HTTP_OK) {InputStream inputStream = connection.getInputStream();return inputStream;}} catch (IOException e) {e.printStackTrace();}return null;}}

getImageStream这个方法是把线上的文件,转成流

5.2 修改部分:

LicenseVerify文件中,调整证书安装的方法,调用工具类,获取线上文件转换地址

/*** 安装License证书*/public synchronized LicenseContent install(LicenseVerifyParam param){LicenseContent result = null;DateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");try{LicenseManager licenseManager = LicenseManagerHolder.getInstance(initLicenseParam(param));licenseManager.uninstall();InputStream stream = POIUtil.getImageStream(param.getLicensePath());String tempFilePath = RuoYiConfig.getProfile() + "/license_temp.lic";File tempFile = new File(tempFilePath);File file = FileUtil.writeFromStream(stream, tempFile);result = licenseManager.install(file);logger.info(MessageFormat.format("证书安装成功,证书有效期:{0} - {1}",format.format(result.getNotBefore()),format.format(result.getNotAfter())));}catch (Exception e){logger.error("证书安装失败,请检查证书是否过期,证书放置位置不正确等因素!", e);}return result;}

总结

这样就做完了license,感谢源码提供方

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/16646.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Activity启动流程要点

一、Activity启动流程 Activity的启动流程一般是通过调用startActivity或者是startActivityForResult来开始的startActivity内部也是通过调用startActivityForResult来启动Activity&#xff0c;只不过传递的requestCode小于0Activity的启动流程涉及到多个进程之间的通讯这里主…

vue 如果有某个子元素就给父元素加样式,或者通过子元素显示来判断父元素是否显示

有这样一个场景&#xff0c;父元素是一个 div&#xff0c;然后里边有多个子元素&#xff0c;同时父元素上添加了很多样式 需求是&#xff1a;如果有子元素&#xff0c;那么就显示父元素&#xff0c;如果没有一个子元素&#xff0c;则父元素也不显示 代码结构&#xff1a; <…

【vue-5】双向数据绑定v-model及修饰符

单向数据绑定&#xff1a;当数据发生改变时&#xff0c;视图会自动更新&#xff0c;但当用户手动更改input的值&#xff0c;数据不会自动更新&#xff1b; 双向数据绑定&#xff1a;当数据发生改变时&#xff0c;视图会自动更新&#xff0c;但当用户手动更改input的值&#xf…

鸿蒙原生应用元服务开发-WEB跨应用跳转

Web组件可以实现点击前端页面超链接跳转到其他应用。 在下面的示例中&#xff0c;点击call.html前端页面中的超连接&#xff0c;跳转到电话应用的拨号界面。 应用侧代码。 // xxx.ets import web_webview from ohos.web.webview; import call from ohos.telephony.call;Entr…

Vue基础(数据绑定、export使用)

1、简介 在使用vue开发的过程中&#xff0c;经常会遇到一些容易混淆的问题&#xff0c;因此&#xff0c;在本文中进行汇总操作&#xff0c;只有通过不断总结学习&#xff0c;才能更好掌握vue的使用&#xff08;每天进步一点&#xff09;。 2、数据绑定 在js中定义数据&#xf…

音乐编曲软件哪个好用 studio one和fl studio哪个好

编曲软件的出现&#xff0c;打破了时间与空间的限制&#xff0c;使得创作者能随时随地进行音乐创作。随着信息时代的发展&#xff0c;使用编曲软件进行音乐创作已经成为业界主流。业内常用的有Cubsae、LogicPro、Studio One、Ableton live等&#xff0c;这次教程我将为大家解读…

HTTP 协议的基本格式和Fidder的简单使用

HTTP协议诞生于1996&#xff08;开玩笑哈&#xff0c;诞生于1991年&#xff09;&#xff0c;http协议用于网页和手机app和服务器交互的场景。通过HTTP协议&#xff0c;客户端&#xff08;例如网页浏览器或手机应用&#xff09;可以向服务器发送请求&#xff0c;服务器则会响应这…

大数据开发面试题【Hadoop篇】

1、Hadoop特点 hadoop是一个分布式计算平台&#xff0c;能够允许使用编程模型在集群上对大型数据集进行分布式处理 hadoop的三大组件&#xff1a;HDFS&#xff08;分布式文件存储平台&#xff09;、MR&#xff08;计算引擎&#xff09;、YARN&#xff08;资源调度平台&#xf…

苹果手机突然白屏无反应怎么办?白屏修复办法分享!

苹果手机突然白屏无反应怎么办&#xff1f;下面小编就来给大家分享苹果手机突然白屏的原因和修复办法。 一般造成苹果手机出现白屏的原因如下&#xff1a; 系统问题&#xff1a;iOS系统的故障是导致苹果设备白屏无反应最常见的原因之一。例如&#xff0c;系统更新失败、应用冲…

TI_DSP_F2808学习笔记3: ePWM

共有6组ePWM&#xff0c;每一组 ePWM 模块都包含以下 7 个模块&#xff1a;时基模块 TB、计数比较模块 CC、动作模块 AQ、死区产生模块 DB、PWM 斩波模块 PC、错误联防模块 TZ、时间触发模块 ET。 时基模块 TB 确定PWM的周期和相位。 1&#xff09;PWM 时基计数器&#xff…

R18 NTN中的RACH-less HO

在看R18 38.300时,发现NTN场景 增加了如下黄色字体的内容,R18 NTN支持了RACH-less HO,索性就简单看了看。 NTN RACH less HO相关的描述主要在38.331,38.213和38.321中。38.300中的描述显示:网络侧会通过RRCReconfiguration消息将RACH-less HO相关的配置下发给UE, 其中会包…

【SpringCloud】负载均衡

目录 负载均衡什么是负载均衡生活场景为什么需要负载均衡负载均衡手段负载均衡总的来说有两种实现手段负载均衡具体可以通过多种手段来实现 SpringCloud中的负载均衡组件Ribbon VS Nginx负载均衡区别集中式LB进程内LB RibbonRibbon的工作原理Ribbon在工作时分成两步 使用1.提供…

光纤跳纤,这篇文章值得一看

光纤跳线作为光网络布线最基础的元件之一&#xff0c;被广泛应用于光纤链路的搭建中。 如今&#xff0c;光纤制造商根据应用场景的不同推出众多类型的光纤跳线&#xff0c;如 MPO / LC / SC / FC / ST 光纤跳线&#xff0c;单工/双工光纤跳线&#xff0c;单模/多模光纤跳线等&…

把maven本地库(windows)导入Nexus3(ubuntu)

1、在nexus中创建导入仓库 点“Create repository” 选择maven2(hosted) 填上对应的仓库name&#xff0c;Version policy选“Mixed” Hosted中的Deployment policy选择“Allow redeploy” 点“Create repository”创建仓库 创建好的仓库如下 记下仓库的url&#xff0c;下…

智慧教室课堂-专注度及考试作弊系统、课堂动态点名,情绪识别、表情识别和人脸识别结合

课堂专注度分析&#xff1a; 课堂专注度表情识别 作弊检测&#xff1a; 关键点计算方法 转头(probe)低头(peep)传递物品(passing) 侧面的传递物品识别 逻辑回归关键点 使用&#xff1a; 运行setup.py安装必要内容 python setup.py build develop 运行demo_inference.py 将…

ISO 27701-2019 隐私信息管理体系要求中文版 学习笔记

ISO 27701-2019: 隐私信息管理体系的新里程碑 随着数字化时代的到来&#xff0c;个人隐私保护变得愈发重要。组织和企业在处理个人数据时&#xff0c;如何确保隐私安全&#xff0c;成为了全球关注的话题。ISO 27701-2019标准&#xff0c;作为隐私信息管理体系&#xff08;PIMS&…

计算机的存储体系

计算机的存储分为内存和硬盘两大类。其中内存属于非持久化的存储设备&#xff0c;用于临时存储数据&#xff0c;设备掉电后数据会丢失&#xff1b;硬盘属于持久化的存储设备&#xff0c;设备掉电后数据不会丢失。 实际上在计算机领域存储的种类是非常多的&#xff0c;业界有时…

osgearth 3.5 vs 2019编译

下载源码 git clone --recurse-submodules https://github.com/gwaldron/osgearth.git 修改配置文件 主要是修改bootstrap_vcpkg.bat&#xff0c;一处是vs的版本&#xff0c;第二处是-DCMAKE_BUILD_TYPERELEASE 构建 执行bootstrap_vcpkg.bat vs中生成安装 vs2019打开bu…

用友电子凭证解决方案,加速企业电子凭证全链路管理

2023年&#xff0c;财政部等9部委联合推进电子凭证数据标准及试点工作&#xff0c;目前逐步扩大试点范围&#xff0c;覆盖市场应用高频的9类凭证。2024年&#xff0c;财政部办公厅发布了《关于继续开展电子凭证会计数据标准深化试点工作的通知》对电子凭证进行全流程常态化处理…

赛事|基于SprinBoot+vue的CSGO赛事管理系统(源码+数据库+文档)

CSGO赛事管理系统 目录 基于SprinBootvue的CSGO赛事管理系统 一、前言 二、系统设计 三、系统功能设计 1系统功能模块 2管理员功能模块 3参赛战队功能模块 4合作方功能模块 四、数据库设计 五、核心代码 六、论文参考 七、最新计算机毕设选题推荐 八、源码获取&…