审计日志功能实现优化及测试记录(参照若依系统,以dolphinscheduler 2.0.5 为例,实现相关功能)

news2025/1/30 16:31:53

目录

  • 🐬使用
    • 🐠若依-操作日志
    • 🐠引入海豚调度
      • 🐟引入审计日志包,增加`LogAnnotation`注解
  • 🐬问题记录及优化
    • 🐠service方法注解时而生效,时而不生效
      • 🐟不生效原因
      • 🐟修改
        • 🐡自我注入(纯测试)
        • 🐡接口中增加该方法
    • 🐠优化,增加批次号
      • 🐟ThreadLocal的使用
        • 🐡测试结果
      • 🐟地理位置的获取

前段时间调研审计日志,发现海豚调度3.0源码已经有这个功能了,测试了一下发现功能并没有完全实现,详见dolphinscheduler 3.0.1 审计日志,至于2.0的版本很早就有审计日志相关类了,只不过都是开发未完成状态。目前是参照若依管理系统的操作日志功能,把代码提取出来之后,稍作优化,本文主要记录做了哪些优化以、使用过程遇到的注意事项及相关技术点。

🐬使用


🐠若依-操作日志


在这里插入图片描述
在这里插入图片描述

🐠引入海豚调度


直接提取若依中记录操作日志的相关类,运用的AOP,切面、切点。核心类为LogAspectLog,提取出来之后,根据报错,缺少哪个类,把相关类拷贝过来即可,我这边没有直接放到海豚调度,而是单独搞了个项目,主要为了方便其他项目也能调用。

  • LogAspect:切面处理类,解析请求响应参数入库等

    com.ruoyi.framework.aspectj.LogAspect
    
  • 切点注解接口,拎出来之后,接口名称我改成了LogAnnotation,海豚2.0自带接口为AccessLogAnnotation,只不过是个开发未完成的功能,由于这部分要做成公共的功能,所以直接放弃了海豚这部分的半成品

    com.ruoyi.common.annotation.Log
    
  • 表为操作日志记录表-sys_oper_log

    CREATE TABLE `sys_oper_log` (
      `oper_id` bigint NOT NULL AUTO_INCREMENT COMMENT '日志主键',
      `title` varchar(50) DEFAULT '' COMMENT '模块标题',
      `business_type` int DEFAULT '0' COMMENT '业务类型(0其它 1新增 2修改 3删除)',
      `method` varchar(100) DEFAULT '' COMMENT '方法名称',
      `request_method` varchar(10) DEFAULT '' COMMENT '请求方式',
      `operator_type` int DEFAULT '0' COMMENT '操作类别(0其它 1后台用户 2手机端用户)',
      `oper_name` varchar(50) DEFAULT '' COMMENT '操作人员',
      `dept_name` varchar(50) DEFAULT '' COMMENT '部门名称',
      `oper_url` varchar(255) DEFAULT '' COMMENT '请求URL',
      `oper_ip` varchar(128) DEFAULT '' COMMENT '主机地址',
      `oper_location` varchar(255) DEFAULT '' COMMENT '操作地点',
      `oper_param` varchar(2000) DEFAULT '' COMMENT '请求参数',
      `json_result` varchar(2000) DEFAULT '' COMMENT '返回参数',
      `status` int DEFAULT '0' COMMENT '操作状态(0正常 1异常)',
      `error_msg` varchar(2000) DEFAULT '' COMMENT '错误消息',
      `oper_time` datetime DEFAULT NULL COMMENT '操作时间',
      PRIMARY KEY (`oper_id`)
    ) ENGINE=InnoDB AUTO_INCREMENT=1068 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='操作日志记录';
    

🐟引入审计日志包,增加LogAnnotation注解


以工作流定义-修改为例,增加注解,原先的AccessLogAnnotation注解直接去掉了,没用,此时编辑工作流的时候,请求响应参数及方法等都会存到数据库
在这里插入图片描述

🐬问题记录及优化


🐠service方法注解时而生效,时而不生效


背景:工作流定义-更新,内部涉及三张表的操作,对应多个方法,因此想要记录所有的方法的请求响应参数

🐟不生效原因


  • 添加注解的方法必须是public,内部私有方法代理是获取不到的
  • 添加注解的方法类必须实例化过
    • 比如工作流定义更新方法,私有方法updateDagDefine,即便改为public,添加注解也不会生效
      在这里插入图片描述

🐟修改


🐡自我注入(纯测试)


改为public,注入本身,再次调用,注解有效(纯测试,虽然spring支持循环依赖,可以自我注入,但是总感觉这样干有风险)
在这里插入图片描述
在这里插入图片描述

🐡接口中增加该方法


比如改成updateProcessDefinition这种,通过controller中注入该方法接口
在这里插入图片描述
在这里插入图片描述

🐠优化,增加批次号


背景:工作流定义-更新,内部涉及三张表的操作,对应多个方法,添加注解后全部记录了下来,但是不好区分,哪些记录是同一个操作产生的,即同一个批次,因此增加批次号-batch_no字段,更新工作流操作,底层涉及三种表的更新,这些操作都是在同一个线程,因此可以通过ThreadLocal去实现
在这里插入图片描述

🐟ThreadLocal的使用


ThreadLocal使用很简单,一个set方法,一个get方法,一个remove方法,主要风险在于内存泄漏,一般调用完get方法需要调用remove,也可以实现AutoCloseable ,自动回收,主要何时调用remove不好辨别

  • 新增批次号工具类BatchNoUtils ,实现了AutoCloseable 接口,否则还要单独调用remove方法
    public class BatchNoUtils  implements AutoCloseable {
    
        private static final ThreadLocal<Long> threadLocal = new ThreadLocal<>();
    
        public static void add(Long batchNo) {
            threadLocal.set(batchNo);
        }
    
        public static Long getBatchNo() {
            return threadLocal.get();
        }
    
        @Override
        public void close() {
            threadLocal.remove();
        }
    }
    
  • 增加过滤器BatchNoFilter ,在调用方法之前设置批次号(即threadLocal.set(),初始化值,直接用的yyyyMMddHHmmssSSS
    import java.io.IOException;
    import javax.servlet.FilterChain;
    import javax.servlet.ServletException;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    import org.springframework.stereotype.Component;
    import org.springframework.web.filter.OncePerRequestFilter;
    import com.copote.comb.manager.util.DateUtils;
    
    /**
     * 请求处理前生成批次号.
     *
     * @author rxz
     */
    @Component
    public class BatchNoFilter extends OncePerRequestFilter {
        @Override
        protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)
                throws ServletException, IOException {
            BatchNoUtils.add(Long.valueOf(DateUtils.dateTimeNow("yyyyMMddHHmmssSSS")));
            filterChain.doFilter(request, response);
        }
    }
    
  • 获取批次号
    在这里插入图片描述

🐡测试结果


在这里插入图片描述

🐟地理位置的获取


测试结果中,可以看到IP归属地,北京市,看了下代码,直接调用http://whois.pconline.com.cn/ipJson.jsp,联网就行,本来以为这种操作都是收费的呢,看来也有免费的接口
在这里插入图片描述
在这里插入图片描述

  • 测试类
package com.renxiaozhao.demo;

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.net.URL;
import java.net.URLConnection;

public class Test2 {

	public static void main(String[] args) {
	    getLocation("192.168.3.10");
	    getLocation("36.99.142.234");
    }
	private static String getLocation(String ip) {
	    StringBuilder result = new StringBuilder();
        BufferedReader in = null;
        String url = "http://whois.pconline.com.cn/ipJson.jsp?";
        try {
            String param = "ip=" + ip + "&json=true";
            URL realUrl = new URL(url+param);
            URLConnection connection = realUrl.openConnection();
            connection.setRequestProperty("accept", "*/*");
            connection.setRequestProperty("connection", "Keep-Alive");
            connection.setRequestProperty("user-agent", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1;SV1)");
            connection.connect();
            in = new BufferedReader(new InputStreamReader(connection.getInputStream(), "GBK"));
            String line;
            while ((line = in.readLine()) != null) {
                result.append(line);
            }
            System.out.println(result);
        } catch (Exception e) {
            System.out.println("获取IP归属地异常:" + e.getMessage());
        } finally {
            try {
                if (in != null) {
                    in.close();
                }
            } catch (Exception ex) {
                System.out.println("关闭流异常:" + ex.getMessage());
            }
        }
        return result.toString();
	}
}

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

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

相关文章

基于springboot+mybatis+sqlserver+jsp运行会报名管理系统

基于springbootmybatissqlserverjsp运行会报名管理系统一、系统介绍二、功能展示1.用户登陆2.用户注册3.项目列表(运行员)4.报名列表&#xff08;运动员&#xff09;5.运动员项目列表&#xff08;管理员&#xff09;6.添加项目四、获取源码一、系统介绍 系统主要功能&#xff…

【计算机组成原理Note】2.4.2 加法器

文章目录加法器1. 一位全加器2. 串行加法器3. 串行进位加法器加法器4. 先行进位加法器加法器 1. 一位全加器 一位全加器&#xff0c;当前位输出&#xff1a;SiAi⨁Bi⨁Ci也就是输入中有奇数个1时输出为1&#xff08;异或&#xff09;向高位的进位&#xff1a;CiAiBi(Ai⨁Bi)Ci…

【目标检测】LLA: Loss-aware label assignment for dense pedestrian detection【标签分配】

总结 本文提出了一种用于行人目标检测的标签分配策略&#xff0c;具体来说&#xff0c;主要有以下几步流程。 构建代价矩阵。通过网络的前向传播得到网络的输出&#xff0c;CclsC^{cls}Ccls, CregC^{reg}Creg&#xff0c;构建代价矩阵CCclsλ∗CregCC^{cls}\lambda*C^{reg}CC…

小程序环境切换自定义组件

背景&#xff1a; 最近一直有参与小程序的项目&#xff0c;发现切换环境时经常要上传然后再设置为体验版&#xff0c;比较麻烦&#xff0c;所以尝试做了个切换环境的组件&#xff0c;分享给大家&#xff0c;希望大家能用得上&#xff0c;提点建议 组件长这个样子 展开后 功能&a…

JVM的垃圾回收机制(GC)

系列文章目录 JVM的内存区域划分_crazy_xieyi的博客-CSDN博客 JVM类加载&#xff08;类加载过程、双亲委派模型&#xff09;_crazy_xieyi的博客-CSDN博客 文章目录 一、什么是垃圾回收&#xff1f;二、java的垃圾回收&#xff0c;要回收的内存是哪些&#xff1f;三、回收堆上…

FPGA Base Xilinx跨时钟域宏XPM_CDC

FPGA Base Xilinx跨时钟域宏XPM_CDC最近看手底下的小伙子们写代码&#xff0c;对于跨时钟域的处理极度的不规范&#xff0c;还是放下这句话基础不牢&#xff0c;地动山摇 其实Xilinx公司已经为用户提供了宏定义&#xff0c;实现跨时钟域处理&#xff0c;见截图 XPM_CDC在命名上…

关于旅游景点主题的HTML网页设计——青岛民俗 7页 带登录注册

⛵ 源码获取 文末联系 ✈ Web前端开发技术 描述 网页设计题材&#xff0c;DIVCSS 布局制作,HTMLCSS网页设计期末课程大作业 | 游景点介绍 | 旅游风景区 | 家乡介绍 | 等网站的设计与制作| HTML期末大学生网页设计作业 HTML&#xff1a;结构 CSS&#xff1a;样式 在操作方面上运…

【C++】简化源码——vector的模拟实现

文章目录一、前言二、无参构造&析构三、基础接口1.empty和clear2.size和capacity3.[]和iterator四、resize和reserve五、尾插尾删六、其他构造七、迭代器失效1.insert2.erase八、memcpy问题九、vector.h一、前言 本篇的目的很简单&#xff0c;只有一个&#xff1a;模拟实现…

C语言刷题(一)

&#x1f412;博客名&#xff1a;平凡的小苏 &#x1f4da;学习格言&#xff1a;别人可以拷贝我的模式&#xff0c;但不能拷贝我不断往前的激情 目录 用递归法求一个整数一维数组a的最大元素 猴子吃桃问题 奇偶数换位问题 水仙花数&#xff08;0-100000&#xff09; 换啤酒…

web前端电影项目作业源码 大学生影视主题网页制作电影网页设计模板 学生静态网页作业成品 dreamweaver电影HTML网站制作

HTML实例网页代码, 本实例适合于初学HTML的同学。该实例里面有设置了css的样式设置&#xff0c;有div的样式格局&#xff0c;这个实例比较全面&#xff0c;有助于同学的学习,本文将介绍如何通过从头开始设计个人网站并将其转换为代码的过程来实践设计。 文章目录一、网页介绍一…

redis命令行操作库、键、和五大数据类型详解

一、数据库操作命令 redis默认有16个数据库&#xff0c;类似数组下标从0开始&#xff0c;初始默认使用0号库。 1.1 测试是否连通 ping测试服务器是否连通 返回pone就是连通了 1.2 切换数据库 select index1.3 数据移动 move key db1.4 显示数据总量 dbsize1.5 数据清除 …

Maven 跳过测试的几种方式

在 Maven 对项目进行编译的时候&#xff0c;我们通常可能会希望跳过复杂的测试。 尤其是在开始项目还不是非常稳定的阶段。 命令行中使用 -Dmaven.test.skiptrue 在命令行&#xff0c;只要简单的给任何目标添加 maven.test.skip 属性就能跳过测试&#xff1a; mvn install …

leetcode:6251. 统计回文子序列数目【dp + 统计xy子序列出现的个数】

目录题目截图题目分析ac code总结题目截图 题目分析 固定了中间的数i后从两边选xy 和 yx对于x y的情况&#xff0c;比较简单预处理每个数字出现的index为ids然后看看两边x各自的个数n1 n2n1和n2必须大于等于2左边可以选n1 * (n1 - 1) // 2右边可以选n2 * (n2 - 1) // 2两边乘…

【C++】通过哈希表实现map和set

前言 在前面&#xff0c;我们通过红黑树这一底层结构实现了map和set。它们是关联式容器。而现在&#xff0c;我们将通过哈希表这一数据结构重新实现map和set&#xff0c;即unordered系列的关联式容器。因为它们的遍历是无序的&#xff0c;和平衡二叉树不同&#xff0c;不能做到…

APOLLO UDACITY自动驾驶课程笔记——规划、控制

1、路径规划使用三个输入&#xff0c;第一个输入为地图&#xff0c;Apollo提供的地图数据包括公路网和实时交通信息。第二个输入为我们当前在地图上的位置。第三个输入为我们的目的地&#xff0c;目的地取决于车辆中的乘客。 2、将地图转为图形 该图形由“节点”(node)和“边缘…

直流潮流计算matlab程序

一、直流潮流计算原理 直流潮流发的特点是用电力系统的交流潮流&#xff08;有功功率和无功功率&#xff09;等值的直流电流来代替。甚至只用直流电路的解析法来分析电力系统的有功潮流&#xff0c;而不考虑无功分布对有功的影响。这样一来计算速度加快&#xff0c;但计算的准确…

Rocket MQ : 拒绝神化零拷贝

注: 本文绝非对零拷贝机制的否定笔者能力有限&#xff0c;理解偏差请大家多多指正不可否认零拷贝对于Rocket MQ的高性能表现有着积极正面的作用&#xff0c;但是笔者认为只是锦上添花&#xff0c;并非决定性因素。Rocket MQ性能卓越的原因绝非零拷贝就可以一言以蔽之。 笔者企图…

第146篇 笔记-智能合约介绍

定义&#xff1a;当满足某些预定义条件时&#xff0c;智能合约是一种在区块链网络上运行的防篡改程序。 1.什么是智能合约 智能合约是在区块链网络上托管和执行的计算机程序。每个智能合约都包含指定预定条件的代码&#xff0c;这些条件在满足时会触发并产生结果。通过在去中…

IDEA热部署插件JRebel and XRebel

IDEA热部署插件JRebel and XRebel嘚吧嘚下载安装激活配置使用嘚吧嘚 刚开始用过一段时间的eclipse&#xff0c;其他方面没感觉&#xff0c;但是eclipse的热部署真的是深得我心啊&#x1f60a;。 后来换了IDEA&#xff0c;瞬间就心动了&#xff0c;各个方面真的很好用&#xf…

U3D VideoPlayer播放视频和坑点

最近做的游戏里,需要先播放一段几秒钟的工作室LOGO片头,拿到的视频是AVI格式,以前没在U3D里用到过视频,本以为很简单,没想到都2022年了,U3D播放视频还这么烂。。。 插件最好用的是AVPro,除非你有大量的视频要播放,否则没必要用插件,一个是贵,另一个插件很大。 首先…