spring security获取用户信息的实现代码

作者:蔺荆门 时间:2022-09-29 01:59:03 

前言

我们在使用spring security的时候可以通过好几种方法获取用户信息, 但是今天这篇文章介绍的是一个笔者觉得最优雅的实现; 借鉴现有的spring security controller自动注入参数的方法, 我们来进一步的实现更适合我们业务的用户信息获取方法;

思路

现在spring security会在controller自动注入Authentication/Userdetails等参数, 我们拿到这些对象之后还需要一些处理才可以拿到我们需要的信息, 例如用户ID; 那获取用户ID这个步骤其实可以切片的, 我们直接在controller的参数绑定之前, 获取到我们需要的用户信息, 然后添加到request的param里面, 就可以实现获取用户信息, controller里面使用参数名可以直接接收参数;

少啰嗦, 看代码

首先我们这个功能的实现遇到额第一个障碍就是默认的HttpServletRequest是没有提供修改Parameter的方法的, 那么我们即使获取到用户信息也无法写入request; 解决这个问题就需要自己实现一个HttpServletRequestWrapper, 再使用一个Filter替换原来的request;


import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Map;
import java.util.Vector;

/**
* @author sunhao
* @date create in 2019-12-09 14:39:52
*/
public class UserInfoRequest extends HttpServletRequestWrapper {

private Map<String, String[]> params = new HashMap<>();

/**
  * Constructs a request object wrapping the given request.
  *
  * @param request The request to wrap
  *
  * @throws IllegalArgumentException if the request is null
  */
 public UserInfoRequest(HttpServletRequest request) {

super(request);
   //将参数表,赋予给当前的Map以便于持有request中的参数
   this.params.putAll(request.getParameterMap());
 }

/**
  * 在获取所有的参数名,必须重写此方法,否则对象中参数值映射不上
  */
 @Override
 public Enumeration<String> getParameterNames() {
   return new Vector<>(params.keySet()).elements();
 }

/**
  * 重写getParameter方法
  *
  * @param name 参数名
  * @return 返回参数值
  */
 @Override
 public String getParameter(String name) {
   String[] values = params.get(name);
   if (values == null || values.length == 0) {
     return null;
   }
   return values[0];
 }

@Override
 public String[] getParameterValues(String name) {
   String[] values = params.get(name);
   if (values == null || values.length == 0) {
     return null;
   }
   return values;
 }

/**
  * 增加参数
  *
  * @param name 参数名
  * @param value 参数值
  */
 public void addParameter(String name, Object value) {
   if (value != null) {
     if (value instanceof String[]) {
       params.put(name, (String[]) value);
     } else if (value instanceof String) {
       params.put(name, new String[]{(String) value});
     } else {
       params.put(name, new String[]{String.valueOf(value)});
     }
   }
 }
}

这段代码使用了乐傻驴用户的代码, 在此表示感谢; 然后使用Filter将原有的request替换;


@Component
public class UserInfoFilter extends OncePerRequestFilter {

@Override
 protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {
   filterChain.doFilter(new UserInfoRequest(request), response);
 }
}

现在我们可以获取用户信息然后写入request的parameter了, 这个逻辑是在filter里实现还是在interceptor里实现就看读者自己的想法了; 笔者系统里面有多种用户, 获取用户信息的逻辑有所不同, 所以笔者选择使用interceptor来实现, 可以通过自定义注解来控制注入哪种用户信息;


@Component
public class UserInfoInterceptor implements HandlerInterceptor {

@Override
 public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {

Method method = ((HandlerMethod) handler).getMethod();
   AdminUserInfo adminUserInfo = method.getDeclaredAnnotation(AdminUserInfo.class);
   if (adminUserInfo != null) {

// 获取用户信息的逻辑 自由发挥
     Long userId = ((Admin) ((OAuth2Authentication) request.getUserPrincipal()).getPrincipal()).getId();

// 将用户信息写入request的parameter
     ((UserInfoRequest)request).addParameter("userId", userId);
     return true;
   }

EmployeeUserInfo employeeUserInfo = method.getDeclaredAnnotation(EmployeeUserInfo.class);
   if (employeeUserInfo != null) {
     Long userId = ((Employee) ((OAuth2Authentication) request.getUserPrincipal()).getPrincipal()).getId();
     ((UserInfoRequest)request).addParameter("userId", userId);
     return true;
   }

return true;
 }
}

上面我自己写了两个注解, 这两个注解的代码我就不贴出来了, 写这两个注解完全就是为了注入不同的用户信息; 大家可以各自发挥, 注解不是必须的, 如果大家系统里面只有一种用户或者由于其他原因可以直接注入parameter; 接下来配置interceptor


@Configuration
public class WebMvcConfig implements WebMvcConfigurer {

private final UserInfoInterceptor userInfoInterceptor;

public WebMvcConfig(UserInfoInterceptor userInfoInterceptor) {

this.userInfoInterceptor = userInfoInterceptor;
 }

@Override
 public void addInterceptors(InterceptorRegistry registry) {

registry.addInterceptor(userInfoInterceptor);
 }
}

代码写到这里功能已经做完了, 我们可以在controller里面这样获取用户信息


@EmployeeUserInfo // 自定义注解
@GetMapping
public void testObtainUserInfo(Long userId) {

System.out.println("userId = " + userId);
}

写EmployeeUserInfo注解注入的就是employee的用户信息, 写AdminUserInfo注解注入的就是admin的用户信息

来源:https://www.jianshu.com/p/383e13a4e03e

标签:spring,security,用户信息
0
投稿

猜你喜欢

  • Java @Deprecated注解的作用及传递性

    2023-08-11 12:55:05
  • Android音频开发之SurfaceView的使用详解

    2023-01-15 13:56:43
  • C#中判断本地系统的网络连接状态的方法

    2023-07-02 15:39:41
  • Android 读取文件内容实现方法总结

    2022-09-15 12:57:56
  • springboot @Async 注解如何实现方法异步

    2023-11-18 18:16:03
  • Android 判断日期是否在一年以内的算法实例

    2023-03-22 06:04:13
  • C# 使用SpecFlow创建BDD测试用例的示例代码

    2021-05-25 21:35:00
  • Android AIDL实现与服务相互调用方式

    2021-08-09 10:57:40
  • SpringBoot2使用JTA组件实现基于JdbcTemplate多数据源事务管理(亲测好用)

    2021-10-12 06:36:09
  • npoi2.0将datatable对象转换为excel2007示例

    2021-07-15 12:47:30
  • C#实现航班预订系统

    2021-12-26 02:19:37
  • C#定义并实现单链表实例解析

    2022-04-15 12:53:09
  • Android开发实现ListView异步加载数据的方法详解

    2022-03-26 02:31:57
  • 如何基于Java实现对象List排序

    2022-01-28 03:59:16
  • android端微信支付V3版本地签名统一下单详解

    2023-05-18 02:37:44
  • SpringCloud Gateway HttpWebHandlerAdapter链路调用请求流程介绍

    2023-04-29 00:12:18
  • java rocketmq--消息的产生(普通消息)

    2023-10-19 08:51:50
  • 非常实用的小功能 Android应用版本的更新实例

    2023-10-28 03:41:51
  • java用接口、多态、继承、类计算三角形和矩形周长及面积的方法

    2021-10-24 22:15:46
  • 详解Android 全局弹出对话框SYSTEM_ALERT_WINDOW权限

    2023-06-28 04:07:44
  • asp之家 软件编程 m.aspxhome.com