springcloud微服务基于redis集群的单点登录

yls

2019-9-23


简介

本文介绍微服务架构中如何实现单点登录功能

创建三个服务:

  1. 操作redis集群的服务,用于多个服务之间共享数据
  2. 统一认证中心服务,用于整个系统的统一登录认证
  3. 服务消费者,用于测试单点登录

大体思路:每个服务都设置一个拦截器检查cookie中是否有token,若有token,则放行,若没有token,重定向到统一认证中心服务进行登录,登录成功后返回到被拦截的服务。

搭建redis集群服务

搭建redis集群参考文档

搭建统一认证中心

  • 主函数添加注解
/**
* 单点登录既要注册到服务注册中心,又要向redis服务系统获取鼓舞
* 所以要添加 @EnableDiscoveryClient @EnableEurekaClient 两个注解
*
*/ @EnableDiscoveryClient
@EnableEurekaClient
@EnableFeignClients
@MapperScan(basePackages = "com.example.itokenservicesso.mapper")
@SpringBootApplication
public class ItokenServiceSsoApplication { public static void main(String[] args) {
SpringApplication.run(ItokenServiceSsoApplication.class, args);
} }
  • 消费redis服务和熔断器
@FeignClient(value = "itoken-service-redis", fallback = RedisServiceFallBack.class)
public interface RedisService { @PostMapping(value = "put")
public String put(@RequestParam(value = "key") String key, @RequestParam(value = "value") String value, @RequestParam(value = "seconds") long seconds); @GetMapping(value = "get")
public String get(@RequestParam(value = "key") String key); }
@Component
public class RedisServiceFallBack implements RedisService {
@Override
public String put(String key, String value, long seconds) {
return FallBack.badGateWay();
} @Override
public String get(String key) {
return FallBack.badGateWay();
}
}
public class FallBack {

    public static String badGateWay(){
try {
return JsonUtil.objectToString(ResultUtil.error(502,"内部错误"));
} catch (JsonProcessingException e) {
e.printStackTrace();
}
return null;
}
}
  • 登录服务
@Service
public class LoginServiceImpl implements LoginService { @Autowired
private UserMapper userMapper; @Autowired
private RedisService redisService; @Override
public User login(String loginCode, String plantPassword) {
//从缓存中获取登录用户的数据
String json = redisService.get(loginCode); User user = null;
//如果缓存中没有数据,从数据库取数据
if (json == null) {
user = userMapper.selectAll(loginCode);
String passwordMd5 = DigestUtils.md5DigestAsHex(plantPassword.getBytes());
if (user != null && passwordMd5.equals(user.getPassword())) {
//登录成功,刷新缓存
try {
redisService.put(loginCode, JsonUtil.objectToString(user), 60 * 60 * 24);
} catch (JsonProcessingException e) {
e.printStackTrace();
}
return user;
} else {
return null;
}
}
//如果缓存中有数据
else {
try {
user = JsonUtil.stringToObject(json, User.class);
} catch (IOException e) {
e.printStackTrace();
}
}
return user;
}
}
  • contoller层,处理登录业务和登录跳转

    • 登录业务

          /**
      * 登录业务
      *
      * @param loginCode
      * @param password
      * @return
      */
      @PostMapping("login")
      public String login(String loginCode,
      String password,
      @RequestParam(required = false) String url,
      HttpServletRequest request,
      HttpServletResponse response,
      RedirectAttributes redirectAttributes) {
      User user = loginService.login(loginCode, password);
      //登录成功
      if (user != null) { String token = UUID.randomUUID().toString();
      //将token放入缓存
      String result = redisService.put(token, loginCode, 60 * 60 * 24);
      //如果redisService没有熔断,也就是返回ok,才能执行
      if (result != null && result.equals("ok")) {
      CookieUtil.setCookie(response, "token", token, 60 * 60 * 24);
      if (url != null && !url.trim().equals(""))
      return "redirect:" + url;
      }
      //熔断后返回错误提示
      else {
      redirectAttributes.addFlashAttribute("message", "服务器异常");
      } }
      //登录失败
      else {
      redirectAttributes.addFlashAttribute("message", "用户名或密码错误");
      }
      return "redirect:/login";
      }
    • 登录跳转

        @Autowired
    private LoginService loginService; @Autowired
    private RedisService redisService; /**
    * 跳转登录页
    */
    @GetMapping("login")
    public String login(HttpServletRequest request,
    Model model,
    @RequestParam(required = false) String url
    ) {
    String token = CookieUtil.getCookie(request, "token");
    //token不为空可能已登录,从redis获取账号
    if (token != null && token.trim().length() != 0) {
    String loginCode = redisService.get(token);
    //如果账号不为空,从redis获取该账号的个人信息
    if (loginCode != null && loginCode.trim().length() != 0) {
    String json = redisService.get(loginCode);
    if (json != null && json.trim().length() != 0) {
    try {
    User user = JsonUtil.stringToObject(json, User.class); //已登录
    if (user != null) {
    if (url != null && url.trim().length() != 0) {
    return "redirect:" + url;
    }
    }
    //将登录信息传到登录页
    model.addAttribute("user", user); } catch (IOException e) {
    e.printStackTrace();
    } }
    }
    }
    return "login";
    }
  • 搭建服务消费者:添加一个拦截器,判断token是否为空

    • 拦截器
    public class WebAdminInterceptor implements HandlerInterceptor {
    
        @Autowired
    private RedisService redisService; @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
    String token = CookieUtil.getCookie(request, "token"); //token为空,一定没有登录
    if (token == null || token.isEmpty()) {
    response.sendRedirect("http://localhost:8503/login?url=http://localhost:8601/login");
    return false;
    }
    return true;
    } @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { HttpSession session = request.getSession();
    User user = (User) session.getAttribute("user"); //已登陆状态
    if (user != null) {
    if (modelAndView != null) {
    modelAndView.addObject("user", user); }
    }
    //未登录状态
    else {
    String token = CookieUtil.getCookie(request, "token");
    if (token != null && !token.isEmpty()) {
    String loginCode = redisService.get(token); if (loginCode != null && !loginCode.isEmpty()) {
    String json = redisService.get(loginCode);
    if (json != null && !json.isEmpty()) {
    //已登录状态,创建局部会话
    user = JsonUtil.stringToObject(json, User.class);
    if (modelAndView != null) {
    modelAndView.addObject("user", user);
    }
    request.getSession().setAttribute("user", user);
    }
    }
    }
    } //二次确认是否有用户信息
    if (user == null) {
    response.sendRedirect("http://localhost:8503/login?url=http://localhost:8601/login"); } } @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { }
    }
    • 配置拦截器
    @Configuration
    public class WebAdminInterceptorConfig implements WebMvcConfigurer { //将拦截器设置为Bean,在拦截其中才能使用@AutoWired注解自动注入
    @Bean
    WebAdminInterceptor webAdminInterceptor() {
    return new WebAdminInterceptor();
    } @Override
    public void addInterceptors(InterceptorRegistry registry) {
    registry.addInterceptor(webAdminInterceptor())
    .addPathPatterns("/**")
    .excludePathPatterns("/static");
    }
    }
  • 任意写一个接口,触发拦截器进行测试

@RequestMapping(value = {"/login"})
public String index(){
return "index";
}

springcloud微服务基于redis集群的单点登录的更多相关文章

  1. 庐山真面目之六微服务架构Consul集群、Ocelot网关集群和Nginx版本实现

    庐山真面目之六微服务架构Consul集群.Ocelot网关集群和Nginx版本实现 一.简介      在上一篇文章<庐山真面目之五微服务架构Consul集群.Ocelot网关和Nginx版本实 ...

  2. 庐山真面目之七微服务架构Consul集群、Ocelot网关集群和IdentityServer4版本实现

    庐山真面目之七微服务架构Consul集群.Ocelot网关集群和IdentityServer4版本实现 一.简介      在上一篇文章<庐山真面目之六微服务架构Consul集群.Ocelot网 ...

  3. 交付Dubbo微服务到kubernetes集群

    1.基础架构 1.1.架构图 Zookeeper是Dubbo微服务集群的注册中心 它的高可用机制和k8s的etcd集群一致 java编写,需要jdk环境 1.2.节点规划 主机名 角色 ip hdss ...

  4. (转)实验文档2:实战交付一套dubbo微服务到kubernetes集群

    基础架构 主机名 角色 ip HDSS7-11.host.com k8s代理节点1,zk1 10.4.7.11 HDSS7-12.host.com k8s代理节点2,zk2 10.4.7.12 HDS ...

  5. 基于redis集群实现的分布式锁,可用于秒杀商品的库存数量管理,有測试代码(何志雄)

    转载请标明出处. 在分布式系统中,常常会出现须要竞争同一资源的情况,本代码基于redis3.0.1+jedis2.7.1实现了分布式锁. redis集群的搭建,请见我的另外一篇文章:<>& ...

  6. 基于redis集群实现的分布式锁,可用于秒杀,定时器。

    在分布式系统中,经常会出现需要竞争同一资源的情况,使用redis可以实现分布式锁. 前提:redis集群已经整合项目,并且可以直接注入JedisCluster使用: @Autowired privat ...

  7. SpringCloud微服务(基于Eureka+Feign+Hystrix+Zuul)

    一.搭建注册中心 1.1.创建一个cloud-service项目 1.2:POM文件依赖 1 <?xml version="1.0" encoding="UTF-8 ...

  8. 8.实战交付一套dubbo微服务到k8s集群(1)之Zookeeper部署

    1.基础架构 主机名 角色 ip HDSS7-11.host.com K8S代理节点1,zk1 10.4.7.11 HDSS7-12.host.com K8S代理节点2,zk2 10.4.7.12 H ...

  9. 实战交付一套dubbo微服务到k8s集群(1)之Zookeeper部署

    基础架构 主机名 角色 IP地址 mfyxw10.mfyxw.com K8S代理节点1,zk1 192.168.80.10 mfyxw20.mfyxw.com K8S代理节点2,zk2 192.168 ...

随机推荐

  1. 告诉你如何回答"线上CPU100%排查"面试问题

    不知道在大家面试中,有没有遇到这个问题: 生产服务器上部署了几个java程序,突然出现了CPU100%的异常告警,你如何定位出问题呢? 这个问题分为两版回答!高调版对不起,我是做研发的,这个问题在生产 ...

  2. django-搭建BBS关键点总结

    0826自我总结 django-搭建BBS关键点总结 一.关于开口子,直接输入url访问文件内容 django自带开了个口子是static文件可以直接访问到 手动开口子 urs.py from dja ...

  3. MySQL 数据库的设计规范

    网址 :http://blog.csdn.net/yjjm1990/article/details/7525811 1.文档的建立日期.所属的单位.2.数据库的命名规范.视图.3.命名的规范:1)避免 ...

  4. [JZOJ5185] 【NOIP2017提高组模拟6.30】tty's sequence

    Description

  5. 控制反转和依赖注入(对IOC,DI理解+案例)

    理解 控制反转说的官方一点就是面向对象编程中的一种设计原则,可以用来减低计算机代码之间的耦合度.其实就是一种设计思想,大概思想就是把设计好的对象交给容器控制,而不是在你内部直接控制. 依赖注入是控制反 ...

  6. Spring Boot入门(一):搭建Spring Boot项目

    从本篇博客开始,我们开始进入Spring Boot的世界,它的出现使Spring的开发变得更加简洁,因此一经推出受到众多程序员的喜爱. 作为Spring Boot系列的第一篇博客,我们先来讲解下如何搭 ...

  7. 浏览器devtools系列(一)

    作为一个web开发人员免不了去和浏览器打交道,前端人员更是如此. 测试人员可能在代码测试的时候容易定位,问题出现在哪里. 不过供桌中常用的可能就是几个,比如前端人员经常看控制面板调试问题,打印后台数据 ...

  8. LeetCode刷题总结-递归篇

    递归是算法学习中很基本也很常用的一种方法,但是对于初学者来说比较难以理解(PS:难点在于不断调用自身,产生多个返回值,理不清其返回值的具体顺序,以及最终的返回值到底是哪一个?).因此,本文将选择Lee ...

  9. MyBatis 概念

    简介 什么是 MyBatis? MyBatis 是一款优秀的持久层框架,它支持定制化 SQL.存储过程以及高级映射.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyB ...

  10. 三、DockerFile 定制属于自己的专属镜像

    前言 上篇文章我们知道了怎么操作镜像和容器,到基础都是从已经存在的镜像开始的,那我们自己怎样搭建一个镜像并使用它呢?接下来就让我们学习使用dockerfile 创建属于自己的镜像吧. dockerfi ...