news 2026/5/9 10:43:09

返利公众号的安全防护体系:Java Spring Security OAuth2+JWT实现用户身份的多端统一认证

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
返利公众号的安全防护体系:Java Spring Security OAuth2+JWT实现用户身份的多端统一认证

返利公众号的安全防护体系:Java Spring Security OAuth2+JWT实现用户身份的多端统一认证

大家好,我是 微赚淘客系统3.0 的研发者省赚客!

在返利类公众号场景中,用户可能通过微信小程序、H5页面或App访问同一套后端服务。为保障账户安全并实现“一次登录、多端通行”,我们基于 Spring Security + OAuth2 + JWT 构建了统一认证中心。本文将聚焦 Java 实现细节,展示如何通过自定义授权服务器、资源服务器及 Token 管理机制,完成高安全性的多端身份认证。

整体架构设计

系统采用 OAuth2 的密码模式(Resource Owner Password Credentials)结合自定义扩展,适用于自有客户端。JWT 作为 Token 载体,包含用户ID、角色、设备类型等信息,并通过 HS512 签名防篡改。所有敏感接口均受 Spring Security 保护。

JWT 工具类与 Token 生成

首先定义 JWT 工具类,用于签发和解析 Token:

packagejuwatech.cn.security.jwt;importio.jsonwebtoken.Claims;importio.jsonwebtoken.Jwts;importio.jsonwebtoken.SignatureAlgorithm;importorg.springframework.beans.factory.annotation.Value;importorg.springframework.stereotype.Component;importjava.util.Date;importjava.util.HashMap;importjava.util.Map;@ComponentpublicclassJwtTokenUtil{@Value("${jwt.secret:juwatech-secret-key-2026}")privateStringsecret;@Value("${jwt.expiration:86400}")// 24小时privateLongexpiration;publicStringgenerateToken(StringuserId,StringclientId){Map<String,Object>claims=newHashMap<>();claims.put("clientId",clientId);claims.put("userId",userId);returnJwts.builder().setClaims(claims).setIssuedAt(newDate()).setExpiration(newDate(System.currentTimeMillis()+expiration*1000)).signWith(SignatureAlgorithm.HS512,secret).compact();}publicClaimsgetClaimsFromToken(Stringtoken){returnJwts.parser().setSigningKey(secret).parseClaimsJws(token).getBody();}publicbooleanvalidateToken(Stringtoken){try{Claimsclaims=getClaimsFromToken(token);return!claims.getExpiration().before(newDate());}catch(Exceptione){returnfalse;}}}

自定义 UserDetailsService

实现UserDetailsService从数据库加载用户信息:

packagejuwatech.cn.security.service;importjuwatech.cn.mapper.UserMapper;importjuwatech.cn.model.User;importorg.springframework.security.core.userdetails.UserDetails;importorg.springframework.security.core.userdetails.UserDetailsService;importorg.springframework.security.core.userdetails.UsernameNotFoundException;importorg.springframework.stereotype.Service;@ServicepublicclassCustomUserDetailsServiceimplementsUserDetailsService{privatefinalUserMapperuserMapper;publicCustomUserDetailsService(UserMapperuserMapper){this.userMapper=userMapper;}@OverridepublicUserDetailsloadUserByUsername(Stringmobile)throwsUsernameNotFoundException{Useruser=userMapper.selectByMobile(mobile);if(user==null){thrownewUsernameNotFoundException("User not found: "+mobile);}returnorg.springframework.security.core.userdetails.User.withUsername(user.getMobile()).password(user.getPassword())// 已加密存储.authorities("ROLE_USER").accountExpired(false).accountLocked(false).credentialsExpired(false).disabled(false).build();}}

OAuth2 授权服务器配置

配置授权服务器,支持密码模式并集成 JWT:

packagejuwatech.cn.security.oauth2;importjuwatech.cn.security.jwt.JwtTokenUtil;importorg.springframework.context.annotation.Bean;importorg.springframework.context.annotation.Configuration;importorg.springframework.security.authentication.AuthenticationManager;importorg.springframework.security.oauth2.config.annotation.configurers.ClientDetailsServiceConfigurer;importorg.springframework.security.oauth2.config.annotation.web.configuration.AuthorizationServerConfigurerAdapter;importorg.springframework.security.oauth2.config.annotation.web.configuration.EnableAuthorizationServer;importorg.springframework.security.oauth2.config.annotation.web.configurers.AuthorizationServerEndpointsConfigurer;importorg.springframework.security.oauth2.config.annotation.web.configurers.AuthorizationServerSecurityConfigurer;importorg.springframework.security.oauth2.provider.token.TokenStore;importorg.springframework.security.oauth2.provider.token.store.JwtAccessTokenConverter;importorg.springframework.security.oauth2.provider.token.store.JwtTokenStore;@Configuration@EnableAuthorizationServerpublicclassAuthorizationServerConfigextendsAuthorizationServerConfigurerAdapter{privatefinalAuthenticationManagerauthenticationManager;privatefinalJwtTokenUtiljwtTokenUtil;publicAuthorizationServerConfig(AuthenticationManagerauthenticationManager,JwtTokenUtiljwtTokenUtil){this.authenticationManager=authenticationManager;this.jwtTokenUtil=jwtTokenUtil;}@Overridepublicvoidconfigure(ClientDetailsServiceConfigurerclients)throwsException{clients.inMemory().withClient("wechat-miniprogram").secret("{noop}mp-secret-2026").authorizedGrantTypes("password","refresh_token").scopes("read","write").accessTokenValiditySeconds(86400).refreshTokenValiditySeconds(604800);}@BeanpublicTokenStoretokenStore(){returnnewJwtTokenStore(jwtAccessTokenConverter());}@BeanpublicJwtAccessTokenConverterjwtAccessTokenConverter(){JwtAccessTokenConverterconverter=newJwtAccessTokenConverter();converter.setSigningKey("juwatech-secret-key-2026");returnconverter;}@Overridepublicvoidconfigure(AuthorizationServerEndpointsConfigurerendpoints){endpoints.tokenStore(tokenStore()).accessTokenConverter(jwtAccessTokenConverter()).authenticationManager(authenticationManager);}@Overridepublicvoidconfigure(AuthorizationServerSecurityConfigurersecurity){security.allowFormAuthenticationForClients().checkTokenAccess("isAuthenticated()");}}

资源服务器与权限校验

在资源服务器中解析 JWT 并构建 SecurityContext:

packagejuwatech.cn.security.resource;importjuwatech.cn.security.jwt.JwtTokenUtil;importorg.springframework.security.authentication.UsernamePasswordAuthenticationToken;importorg.springframework.security.core.context.SecurityContextHolder;importorg.springframework.security.core.userdetails.UserDetails;importorg.springframework.security.core.userdetails.UserDetailsService;importorg.springframework.stereotype.Component;importorg.springframework.web.filter.OncePerRequestFilter;importjavax.servlet.FilterChain;importjavax.servlet.ServletException;importjavax.servlet.http.HttpServletRequest;importjavax.servlet.http.HttpServletResponse;importjava.io.IOException;@ComponentpublicclassJwtAuthenticationFilterextendsOncePerRequestFilter{privatefinalJwtTokenUtiljwtTokenUtil;privatefinalUserDetailsServiceuserDetailsService;publicJwtAuthenticationFilter(JwtTokenUtiljwtTokenUtil,UserDetailsServiceuserDetailsService){this.jwtTokenUtil=jwtTokenUtil;this.userDetailsService=userDetailsService;}@OverrideprotectedvoiddoFilterInternal(HttpServletRequestrequest,HttpServletResponseresponse,FilterChainchain)throwsServletException,IOException{StringauthHeader=request.getHeader("Authorization");if(authHeader!=null&&authHeader.startsWith("Bearer ")){Stringtoken=authHeader.substring(7);if(jwtTokenUtil.validateToken(token)){Stringmobile=jwtTokenUtil.getClaimsFromToken(token).getSubject();UserDetailsuserDetails=userDetailsService.loadUserByUsername(mobile);UsernamePasswordAuthenticationTokenauthentication=newUsernamePasswordAuthenticationToken(userDetails,null,userDetails.getAuthorities());SecurityContextHolder.getContext().setAuthentication(authentication);}}chain.doFilter(request,response);}}

并在 WebSecurity 中启用:

packagejuwatech.cn.security.config;importjuwatech.cn.security.resource.JwtAuthenticationFilter;importorg.springframework.context.annotation.Bean;importorg.springframework.context.annotation.Configuration;importorg.springframework.security.config.annotation.web.builders.HttpSecurity;importorg.springframework.security.config.annotation.web.configuration.EnableWebSecurity;importorg.springframework.security.config.http.SessionCreationPolicy;importorg.springframework.security.web.SecurityFilterChain;importorg.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter;@Configuration@EnableWebSecuritypublicclassResourceServerConfig{privatefinalJwtAuthenticationFilterjwtAuthenticationFilter;publicResourceServerConfig(JwtAuthenticationFilterjwtAuthenticationFilter){this.jwtAuthenticationFilter=jwtAuthenticationFilter;}@BeanpublicSecurityFilterChainfilterChain(HttpSecurityhttp)throwsException{http.csrf().disable().sessionManagement().sessionCreationPolicy(SessionCreationPolicy.STATELESS).and().authorizeRequests().antMatchers("/api/auth/**").permitAll().antMatchers("/api/refund/**").hasRole("USER").anyRequest().authenticated().and().addFilterBefore(jwtAuthenticationFilter,UsernamePasswordAuthenticationFilter.class);returnhttp.build();}}

多端统一登录接口

提供统一登录入口,适配不同客户端:

packagejuwatech.cn.controller;importjuwatech.cn.security.jwt.JwtTokenUtil;importorg.springframework.security.authentication.AuthenticationManager;importorg.springframework.security.authentication.UsernamePasswordAuthenticationToken;importorg.springframework.web.bind.annotation.PostMapping;importorg.springframework.web.bind.annotation.RequestBody;importorg.springframework.web.bind.annotation.RestController;@RestControllerpublicclassAuthController{privatefinalAuthenticationManagerauthenticationManager;privatefinalJwtTokenUtiljwtTokenUtil;publicAuthController(AuthenticationManagerauthenticationManager,JwtTokenUtiljwtTokenUtil){this.authenticationManager=authenticationManager;this.jwtTokenUtil=jwtTokenUtil;}@PostMapping("/api/auth/login")publicObjectlogin(@RequestBodyLoginRequestreq){authenticationManager.authenticate(newUsernamePasswordAuthenticationToken(req.getMobile(),req.getPassword()));Stringtoken=jwtTokenUtil.generateToken(req.getMobile(),req.getClientId());returnjava.util.Map.of("access_token",token,"token_type","Bearer");}publicstaticclassLoginRequest{privateStringmobile;privateStringpassword;privateStringclientId;// getters and setterspublicStringgetMobile(){returnmobile;}publicvoidsetMobile(Stringmobile){this.mobile=mobile;}publicStringgetPassword(){returnpassword;}publicvoidsetPassword(Stringpassword){this.password=password;}publicStringgetClientId(){returnclientId;}publicvoidsetClientId(StringclientId){this.clientId=clientId;}}}

本文著作权归 微赚淘客系统3.0 研发团队,转载请注明出处!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/8 10:18:48

【小程序毕设源码分享】基于springboot+小程序的武设专业解读武术兴趣班小程序的设计与实现(程序+文档+代码讲解+一条龙定制)

博主介绍&#xff1a;✌️码农一枚 &#xff0c;专注于大学生项目实战开发、讲解和毕业&#x1f6a2;文撰写修改等。全栈领域优质创作者&#xff0c;博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战 ✌️技术范围&#xff1a;&am…

作者头像 李华
网站建设 2026/5/6 10:49:45

如何科学地“设计”SFT 数据?一次关于 ODA 的完整平台级验证

在大模型后训练阶段&#xff0c;SFT&#xff08;监督微调&#xff09;数据的构建至关重要。然而&#xff0c;长期以来&#xff0c;这一过程业界的通行做法往往依赖“直觉”或“试错”&#xff0c;即多收一点、再筛一轮、训一次模型、看下效果&#xff0c;然后再调整。这个过程不…

作者头像 李华
网站建设 2026/5/9 15:06:18

黑客攻击MongoDB实例删除数据库并植入勒索信息

威胁行为者正通过大规模自动化勒索软件活动&#xff0c;持续攻击暴露在互联网上的MongoDB实例。攻击模式高度一致&#xff1a;攻击者扫描公网可访问的未受保护MongoDB数据库&#xff0c;删除存储数据后植入比特币勒索信息。 MongoDB实例遭入侵分析 最新证据显示&#xff0c;尽…

作者头像 李华
网站建设 2026/5/6 10:51:34

基于SSM的高校旧书交易系统的设计与实现(毕业论文)

摘 要 随着教育资源的日益丰富和高等教育的普及&#xff0c;大学生群体在学习和科研过程中产生了大量的书籍需求。然而&#xff0c;由于课程结束或毕业离校等原因&#xff0c;许多书籍在使用一段时间后便被闲置&#xff0c;这造成了大量资源的浪费。基于此&#xff0c;本文基于…

作者头像 李华
网站建设 2026/5/9 19:38:24

Vue与Web Components的集成:技术原理、实践方案与生态协同

Vue与Web Components的集成&#xff1a;技术原理、实践方案与生态协同 一、技术演进背景与核心价值 Web Components作为W3C标准化的浏览器原生组件技术&#xff0c;由Custom Elements、Shadow DOM和HTML Templates三大核心规范构成。其设计初衷在于解决Web开发中的组件复用难题…

作者头像 李华
网站建设 2026/5/6 11:52:08

GitHub项目上传、删除与协议设置:新手到高手的完整指南

GitHub项目上传、删除与协议设置&#xff1a;新手到高手的完整指南 引言 对于每一位开发者而言&#xff0c;GitHub不仅是代码的托管平台&#xff0c;更是个人技术履历和协作开发的核心。然而&#xff0c;从如何将第一个项目成功推送&#xff0c;到管理项目生命周期&#xff0…

作者头像 李华