传统的登录系统中,每个站点都实现了自己的专用登录模块。各站点的登录状态相互不认可,各站点需要逐一手工登录。例如:
这样的系统,我们又称之为多点登陆系统。应用起来相对繁琐(每次访问资源服务都需要重新登陆认证和授权)。与此同时,系统代码的重复也比较高。由此单点登陆系统诞生。
单点登陆系统概述单点登录,英文是 Single Sign On(缩写为 SSO)。即多个站点共用一台认证授权服务器,用户在其中任何一个站点登录后,可以免登录访问其他所有站点。而且,各站点间可以通过该登录状态直接交互。例如:
单点登陆系统解决方案设计
- 解决方案1:用户登陆成功以后,将用户登陆状态存储到redis数据库,例如:
说明,在这套方案中,用户登录成功后,会基于UUID生成一个token,然后与用户信息绑定在一起存储到数据库.后续用户在访问资源时,基于token从数据库查询用户状态,这种方式因为要基于数据库存储和查询用户状态,所以性能表现一般.
解决方案2:用户登陆成功以后,将用户信息存储到token(令牌),然后写到客户端进行存储。(本次设计方案)
单点登陆系统初步设计 服务设计说明,在这套方案中,用户登录成功后,会基于JWT技术生成一个token,用户信息可以存储到这个token中.后续用户在访问资源时,对token内容解析,检查登录状态以及权限信息,无须再访问数据库.
基于单点登陆系统中的业务描述,进行初步服务架构设计,如图所示:
其中,服务基于业务进行划分,系统(system)服务只提供基础数据(例如用户信息,日志信息等),认证服务(auth)负责完成用户身份的校验,密码的比对,资源服务(resource)代表一些业务服务(例如我的订单,我的收藏等等).
工程结构设计SSO父工程创建及初始化 创建父工程
第一步:创建父工程,例如
第二步:删除父工程src目录(可选)。
父工程pom文件初始配置初始化pom文件内容,例如:
系统基础服务工程设计及实现 业务描述4.0.0 com.jt 02-sso1.0-SNAPSHOT org.springframework.boot spring-boot-dependencies2.3.2.RELEASE pom import org.springframework.cloud spring-cloud-dependenciesHoxton.SR9 pom import com.alibaba.cloud spring-cloud-alibaba-dependencies2.2.6.RELEASE pom import org.projectlombok lombokprovided org.springframework.boot spring-boot-starter-testtest org.junit.jupiter junit-jupiter-engineorg.apache.maven.plugins maven-compiler-plugin3.8.1 8
本次设计系统服务(System),主要用于提供基础数据服务,例如日志信息,用户信息等。
表结构设计系统服务模块,基本表结构设计,例如:
工程数据初始化
将jt-sso.sql文件在mysql中执行一下,其过程如下:
第一:登录mysql
mysql -uroot -proot
第二:通过source指令执行jt-sso.sql文件
创建系统服务工程并初始化source d:/jt-sso.sql
第一步:创建sso-system工程,例如:
第二步:添加项目依赖,例如
mysql mysql-connector-javacom.baomidou mybatis-plus-boot-starter3.4.2 com.alibaba.cloud spring-cloud-starter-alibaba-nacos-discoverycom.alibaba.cloud spring-cloud-starter-alibaba-nacos-configorg.springframework.boot spring-boot-starter-web
第三步:在项目中添加bootstrap.yml文件,其内容如下:
server: port: 8061 spring: application: name: sso-system cloud: nacos: discovery: server-addr: localhost:8848 config: server-addr: localhost:8848 file-extension: yml datasource: url: jdbc:mysql:///jt-sso?serverTimezone=Asia/Shanghai&characterEncoding=utf8 username: root password: root
第三步:在项目中添加bootstrap.yml文件,其内容如下:
server: port: 8061 spring: application: name: sso-system cloud: nacos: discovery: server-addr: localhost:8848 config: server-addr: localhost:8848 file-extension: yml datasource: url: jdbc:mysql:///jt-sso?serverTimezone=Asia/Shanghai&characterEncoding=utf8 username: root password: root
说明,可将连接数据库的配置,添加到配置中心。
第四步:在项目中添加启动类,例如:
package com.jt; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; @SpringBootApplication public class SystemApplication { public static void main(String[] args) { SpringApplication.run(SystemApplication.class,args); } }
第五步:在项目中添加单元测试类,测试数据库连接,例如:
package com.jt; import org.junit.jupiter.api.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; import javax.sql.DataSource; import java.sql.Connection; import java.sql.SQLException; @SpringBootTest public class DataSourceTests { @Autowired private DataSource dataSource;//HikariDataSource @Test void testGetConnection() throws SQLException { Connection conn= dataSource.getConnection(); System.out.println(conn); } }Pojo对象逻辑实现
添加项目User对象,用于封装用户信息。
package com.jt.system.pojo; import lombok.Data; import java.io.Serializable; @Data public class User implements Serializable { private static final long serialVersionUID = 4831304712151465443L; private Long id; private String username; private String password; private String status; }Dao对象逻辑实现
第一步:创建UserMapper接口,并定义基于用户名查询用户信息,基于用户id查询用户权限信息的方法,代码如下:
package com.jt.system.dao; import com.baomidou.mybatisplus.core.mapper.baseMapper; import com.jt.system.pojo.User; import org.apache.ibatis.annotations.Mapper; import org.apache.ibatis.annotations.Select; import java.util.List; @Mapper public interface UserMapper extends baseMapper{ @Select("select id,username,password,status " + "from tb_users " + "where username=#{username}") User selectUserByUsername(String username); @Select("select distinct m.permission " + "from tb_user_roles ur join tb_role_menus rm on ur.role_id=rm.role_id" + " join tb_menus m on rm.menu_id=m.id " + "where ur.user_id=#{userId}") List selectUserPermissions(Long userId); }
第二步:创建UserMapperTests类,对业务方法做单元测试,例如:
package com.jt; import com.jt.system.pojo.User; import com.jt.system.dao.UserMapper; import org.junit.jupiter.api.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; import java.util.List; @SpringBootTest public class UserMapperTests { @Autowired private UserMapper userMapper; @Test void testSelectUserByUsername(){ User user = userMapper.selectUserByUsername("admin"); System.out.println(user); } @Test void testSelectUserPermissions(){ ListService对象逻辑实现permission= userMapper.selectUserPermissions(1L); System.out.println(permission); } }
创建UserService接口及实现泪,定义用户及用户权限查询逻辑,代码如下:
第一步:定义service接口,代码如下:
package com.jt.system.service; import com.jt.system.pojo.User; import java.util.List; public interface UserService { User selectUserByUsername(String username); ListselectUserPermissions(Long userId); }
第二步:定义service接口实现类,代码如下:
package com.jt.system.service.impl; import com.jt.system.dao.UserMapper; import com.jt.system.pojo.User; import com.jt.system.service.UserService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import java.util.List; @Service public class UserServiceImpl implements UserService { @Autowired private UserMapper userMapper; @Override public User selectUserByUsername(String username) { return userMapper.selectUserByUsername(username); } @Override public ListController对象逻辑实现selectUserPermissions(Long userId) { return userMapper.selectUserPermissions(userId); } }
package com.jt.system.controller; import com.jt.system.pojo.User; import com.jt.system.service.UserService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PathVariable; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RestController; import java.util.List; @RestController @RequestMapping("/user/") public class UserController { @Autowired private UserService userService; @GetMapping("/login/{username}") public User doSelectUserByUsername( @PathVariable("username") String username){ return userService.selectUserByUsername(username); } @GetMapping("/permission/{userId}") public List启动服务进行访问测试doSelectUserPermissions( @PathVariable("userId") Long userId){ return userService.selectUserPermissions(userId); } }
启动sso-system工程服务,打开浏览器分别对用户及用户权限信息的获取进行访问测试
- 基于用户名查询用户信息,例如:
基于用户id(这里假设用户id为1)查询用户权限,例如:
统一认证工程设计及实现 业务描述
用户登陆时调用此工程对用户身份进行统一身份认证和授权。
创建工程及初始化第一步:创建sso-auth工程,如图所示
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)