先写一个登录策略接口LoginStrategy
package com.powernode.strategy;
import org.springframework.security.core.userdetails.UserDetails;
/**
* 登录策略接口
*/
public interface LoginStrategy {
/**
* 真正处理登录的方法
* @param username
* @return
*/
UserDetails realLogin(String username);
}
然后,编写不同的接口实现类【都要实现这个策略接口】,有几个登录方式,就写几个接口实现类
eg:普通的前端页面登录
package com.powernode.strategy.impl;
import cn.hutool.core.collection.CollectionUtil;
import cn.hutool.core.util.ObjectUtil;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.powernode.constant.AuthConstants;
import com.powernode.domain.LoginSysUser;
import com.powernode.mapper.LoginSysUserMapper;
import com.powernode.model.SecurityUser;
import com.powernode.strategy.LoginStrategy;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.security.core.userdetails.UserDetails;
import org.springframework.stereotype.Service;
import java.util.Set;
/**
* 商城后台管理系统登录策略的具体实现
* @Service(AuthConstants.SYS_USER_LOGIN) 表示这个实现类对应一个名为 "sys_user_login" 的登录策略,通过策略模式动态选择
* Spring 把当前实现类注册到容器时,以字符串 "sys_user_login" 作为它的 beanName
* Spring 会自动把所有实现类按 beanName -> 实现对象的映射注入Map里,也就是说按照key-value的方式存储到map里,想要类对象时,可以直接通过key到map里去取
*/
@Service(AuthConstants.SYS_USER_LOGIN)
public class SysUserLoginStrategy implements LoginStrategy {
@Autowired
private LoginSysUserMapper loginSysUserMapper;
@Override
public UserDetails realLogin(String username) {
// 根据用户名称查询用户对象
LoginSysUser loginSysUser = loginSysUserMapper.selectOne(new LambdaQueryWrapper<LoginSysUser>()
.eq(LoginSysUser::getUsername, username)
);
/*LoginSysUser loginSysUser = loginSysUserMapper.selectOne(new QueryWrapper<LoginSysUser>()
.eq("username", username)
);*/
if (ObjectUtil.isNotNull(loginSysUser)) {
// 根据用户标识查询用户的权限集合
Set<String> perms = loginSysUserMapper.selectPermsByUserId(loginSysUser.getUserId());
// 创建安全用户对象SecurityUser
SecurityUser securityUser = new SecurityUser();
securityUser.setUserId(loginSysUser.getUserId());
securityUser.setPassword(loginSysUser.getPassword());
securityUser.setShopId(loginSysUser.getShopId());
securityUser.setStatus(loginSysUser.getStatus());
securityUser.setLoginType(AuthConstants.SYS_USER_LOGIN);
// 判断用户权限是否有值
if (CollectionUtil.isNotEmpty(perms) && perms.size() != 0) {
securityUser.setPerms(perms);
}
return securityUser;
}
return null;
}
}
编写完策略接口后,要写一个工厂类生产LoginStrategyFactory,用于生产不同的策略
package com.powernode.factory;
import com.powernode.strategy.LoginStrategy;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import java.util.HashMap;
import java.util.Map;
/**
* 登录策略工厂类
*/
@Component
public class LoginStrategyFactory {
@Autowired
// 所有的策略实现类都是以map的形式存储的,所以后面的loginStrategyMap.get(loginType)方法才能动态调整登录方式
private Map<String,LoginStrategy> loginStrategyMap = new HashMap<>();
/**
* 根据用户登录类型获取具体的登录策略
* @param loginType
* @return
*/
public LoginStrategy getInstance(String loginType) {
return loginStrategyMap.get(loginType);
}
}
调用
if (!StringUtils.hasText(loginType)) {
throw new InternalAuthenticationServiceException("非法登录,登录类型不匹配");
}
// 通过登录策略工厂获取具体的登录策略对象
LoginStrategy instance = loginStrategyFactory.getInstance(loginType);
return instance.realLogin(username);
1168

被折叠的 条评论
为什么被折叠?



