📕开源风云系列
- 🍊本系列将从开源名将若依出发,探究优质开源项目脚手架汲取编程之道。
- 🍉从不分离版本开写到前后端分离版,再到微服务版本,乃至其中好玩的一系列增强Plus操作。
- 🍈希望你具备如下技术栈: - 🍎Spring- 🍍SpringMVC- 🍐Mybatis/Mybatis-plus- 🍅Thymeleaf- 🥝SpringBoot- 🍓Shiro- 🍏SpringSecurity- 🍌SpringCloud- 🍒云服务器相关知识
- 本篇是不分离版第一篇
- 文章同时同步到我的个人站点🍊欢迎来访!: - 🍎国内:https://www.linxiaoqin.netlify.app- 🍏国际:https://www.linxiaoqin.vercel.app
后期会考虑将项目和安全方面的文章迁移至vuepress,找一个好看的主题,视觉效果++!
1、拉取代码
- 进入若依官网:RuoYi
- 进入Gitee,点击
Forked
,就会Forked一份代码到自己的仓库
- 在自己仓库找到RuoYi仓库,然后复制其 URL
- 打开IDEA,点击
从VCS获取
,粘贴上方复制的URL,然后点击克隆
- OK,现在我们就把RuoYi的代码拉到自己本地啦!
2、创建数据库
- 打开
Navicat
,新建数据库kuangstudy_ruoyi_fast
,字符集为utf8mb4
,排序规则为utf8mb4_general_ci
- 右键运行SQL文件,导入刚才拉取代码sql里面的
quartz.sql
、ry_20240112.sql
- 打开
ruoyi-admin/src/main/resources/application-druid.yml
,修改主库数据源
- 运行
ruoyi-admin/src/main/java/com/ruoyi/web/RuoYiApplication.java
- 打开浏览器,输入
http://localhost:80
,输入默认账户密码:admin/admin123
3、项目结构
com.ruoyi
├── common // 工具类
│ └── annotation // 自定义注解
│ └── config // 全局配置
│ └── constant // 通用常量
│ └── core // 核心控制
│ └── enums // 通用枚举
│ └── exception // 通用异常
│ └── json // JSON数据处理
│ └── utils // 通用类处理
│ └── xss // XSS过滤处理
├── framework // 框架核心
│ └── aspectj // 注解实现
│ └── config // 系统配置
│ └── datasource // 数据权限
│ └── interceptor // 拦截器
│ └── manager // 异步处理
│ └── shiro // 权限控制
│ └── web // 前端控制
├── ruoyi-generator // 代码生成(不用可移除)
├── ruoyi-quartz // 定时任务(不用可移除)
├── ruoyi-system // 系统代码
├── ruoyi-admin // 后台服务
├── ruoyi-xxxxxx // 其他模块
4、配置介绍
4.1、服务器配置
4.1.1、如何修改服务器端口
- 在
ruoyi-admin/src/main/java/application.yml
下查看服务端配置
# 开发环境配置server:# 服务器的HTTP端口,默认为80port:80servlet:# 应用的访问路径context-path: /
tomcat:# tomcat的URI编码uri-encoding: UTF-8# 连接数满后的排队数,默认为100accept-count:1000threads:# tomcat最大线程数,默认为200max:800# Tomcat启动初始化的线程数,默认值10min-spare:100
假如将服务器的HTTP端口改为
8090
,应用的访问路径改为
/admin
,则访问浏览器就从
localhost:80
变成
localhost:8090/admin
4.1.2、如何配置tomcat访问日志
- 修改
application.yml
中的server
开发环境配置
tomcat:# 存放Tomcat的日志目录basedir: D:\Code\IDEA\KuangStudy_RuoYi_Fast\logs\tomcat
accesslog:# 开启日志记录enabled:true# 访问日志存放目录directory: logs
- 重启应用登录
- 进入
E:\Code\IDEA\KuangStudy_RuoYi_Fast\logs\tomcat
则可查看到日志
若需要再额外配置日志,请百度:SpringBoot配置tomcat访问日志
4.2、多环境配置
在我们的日常开发中,生产环境的配置和测试环境的配置以及开发环境的配置基本上都是不相同的,每次到部署环境的时候,就需要手动的切换配置文件,如果在切换的过程中一不小心的话就会出错,所以在开发中,一般会搞个配置文件检查的功能,来避免出错,而Spring Boot则充分考虑了这种情况,为开发人员提供了天然的多环境配置支持。
- 配置文件的名字可以是
application-{profile}.yml
- 在配置文件中指定:
spring.profiles.active={profile}
我们可以创建三个配置文件,分别为:
application-test.yml
: 测试环境application-dev.yml
: 开发环境application-prod.yml
: 生产环境
application-test.yml
内容如下,测试端口为81:
# 开发环境配置server:# 服务器的HTTP端口,测试环境为81port:81servlet:# 应用的访问路径context-path: /
tomcat:# tomcat的URI编码uri-encoding: UTF-8# 连接数满后的排队数,默认为100accept-count:1000threads:# tomcat最大线程数,默认为200max:800# Tomcat启动初始化的线程数,默认值10min-spare:100# 数据源配置spring:datasource:type: com.alibaba.druid.pool.DruidDataSource
driverClassName: com.mysql.cj.jdbc.Driver
druid:# 主库数据源master:url: jdbc:mysql://localhost:3306/kuangstudy_ruoyi_fast?useUnicode=true&characterEncoding=utf8&zeroDateTimeBehavior=convertToNull&useSSL=true&serverTimezone=GMT%2B8username: root
password:123456# 从库数据源slave:# 从数据源开关/默认关闭enabled:falseurl:username:password:# 初始连接数initialSize:5# 最小连接池数量minIdle:10# 最大连接池数量maxActive:20# 配置获取连接等待超时的时间maxWait:60000# 配置连接超时时间connectTimeout:30000# 配置网络超时时间socketTimeout:60000# 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒timeBetweenEvictionRunsMillis:60000# 配置一个连接在池中最小生存的时间,单位是毫秒minEvictableIdleTimeMillis:300000# 配置一个连接在池中最大生存的时间,单位是毫秒maxEvictableIdleTimeMillis:900000# 配置检测连接是否有效validationQuery: SELECT 1 FROM DUAL
testWhileIdle:truetestOnBorrow:falsetestOnReturn:falsewebStatFilter:enabled:truestatViewServlet:enabled:true# 设置白名单,不填则允许所有访问allow:url-pattern: /druid/*
# 控制台管理用户名和密码login-username: ruoyi
login-password:123456filter:stat:enabled:true# 慢SQL记录log-slow-sql:trueslow-sql-millis:1000merge-sql:truewall:config:multi-statement-allow:true
application-dev.yml
开发环境同上,只是端口为80application-prod.yml
生产环境同上,只是端口为82
之后在
application.yml
的
spring.profile.active
指定环境即可:
# Spring配置spring:profiles:# active: test# active: dev# active: prodactive: druid
4.2.1、文档块简洁配置
除了上述配置方法外,我们还可以使用文档块配置,更简洁:
- 在
application.yml
添加
---spring:profiles: dev
# 开发环境配置server:# 服务器的HTTP端口,测试环境为81port:81servlet:# 应用的访问路径context-path: /
tomcat:# tomcat的URI编码uri-encoding: UTF-8# 连接数满后的排队数,默认为100accept-count:1000threads:# tomcat最大线程数,默认为200max:800# Tomcat启动初始化的线程数,默认值10min-spare:100---spring:profiles: test
# 测试环境配置server:# 服务器的HTTP端口,测试环境为81port:81servlet:# 应用的访问路径context-path: /
tomcat:# tomcat的URI编码uri-encoding: UTF-8# 连接数满后的排队数,默认为100accept-count:1000threads:# tomcat最大线程数,默认为200max:800# Tomcat启动初始化的线程数,默认值10min-spare:100---spring:profiles: prod
# 生产环境配置server:# 服务器的HTTP端口,生产环境为82port:82servlet:# 应用的访问路径context-path: /
tomcat:# tomcat的URI编码uri-encoding: UTF-8# 连接数满后的排队数,默认为100accept-count:1000threads:# tomcat最大线程数,默认为200max:800# Tomcat启动初始化的线程数,默认值10min-spare:100
4.2.2、jar包指定环境
如果我们将项目打成 jar 包,如何指定环境运行呢?
# 以开发环境运行java-jar RuoYi.jar --spring.profiles.active=dev
# 以测试环境运行java-jar RuoYi.jar --spring.profiles.active=test
# 以生产环境运行java-jar RuoYi.jar --spring.profiles.active=prod
4.3、读取application.yml
在
ruoyi-admin/src/main/resources/application.yml
里面有项目相关配置,我们来看在代码层面是怎么获取的
# 项目相关配置ruoyi:# 名称name: RuoYi
# 版本version: 4.7.8
# 版权年份copyrightYear:2023# 实例演示开关demoEnabled:true# 上传的文件路径 示例( Windows配置D:/ruoyi/uploadPath,Linux配置 /home/ruoyi/uploadPath)profile: D:/ruoyi/uploadPath
# 获取ip地址开关addressEnabled:false# Shiroshiro:user:# 登录地址loginUrl: /login
# 权限认证失败地址unauthorizedUrl: /unauth
# 首页地址indexUrl: /index
# 验证码开关captchaEnabled:true# 验证码类型 math 数字计算 char 字符验证captchaType: math
cookie:# 设置Cookie的域名 默认空,即当前访问的域名domain:# 设置cookie的有效访问路径path: /
# 设置HttpOnly属性httpOnly:true# 设置Cookie的过期时间,天为单位maxAge:30# 设置密钥,务必保持唯一性(生成方式,直接拷贝到main运行即可)Base64.encodeToString(CipherUtils.generateNewKey(128, "AES").getEncoded()) (默认启动生成随机秘钥,随机秘钥会导致之前客户端RememberMe Cookie无效,如设置固定秘钥RememberMe Cookie则有效)cipherKey:session:# Session超时时间,-1代表永不过期(默认30分钟)expireTime:30# 同步session到数据库的周期(默认1分钟)dbSyncPeriod:1# 相隔多久检查一次session的有效性,默认就是10分钟validationInterval:10# 同一个用户最大会话数,比如2的意思是同一个账号允许最多同时两个人登录(默认-1不限制)maxSession:-1# 踢出之前登录的/之后登录的用户,默认踢出之前登录的用户kickoutAfter:falserememberMe:# 是否开启记住我enabled:true
在
ruoyi-common/src/main/java/com/ruoyi/common/config/RuoYiConfig.java
类是全局配置类,类上面有两个注解:
@Component
: 把这个类交给Spring处理@ConfigurationProperties(prefix = "ruoyi")
: 把配置文件application.yml
里面的值映射到类里面,指定前缀是ruoyi
所以要在其他类中读取到这些信息,可以使用如下第一种方式:
- 导入
RuoYiConfig
类,使用RuoYiConfig.xxx
来访问 - 在Spring中注入使用
@AutoWiredprivateRuoYiConfig ruoYiConfig;// xxx// 调用方法
ruoYiConfig.getName();
也可以使用第二种方式:例如
application.yml
里面的shiro配置
# Shiroshiro:user:# 登录地址loginUrl: /login
# 权限认证失败地址unauthorizedUrl: /unauth
# 首页地址indexUrl: /index
# 验证码开关captchaEnabled:true# 验证码类型 math 数字计算 char 字符验证captchaType: math
cookie:# 设置Cookie的域名 默认空,即当前访问的域名domain:# 设置cookie的有效访问路径path: /
# 设置HttpOnly属性httpOnly:true# 设置Cookie的过期时间,天为单位maxAge:30# 设置密钥,务必保持唯一性(生成方式,直接拷贝到main运行即可)Base64.encodeToString(CipherUtils.generateNewKey(128, "AES").getEncoded()) (默认启动生成随机秘钥,随机秘钥会导致之前客户端RememberMe Cookie无效,如设置固定秘钥RememberMe Cookie则有效)cipherKey:session:# Session超时时间,-1代表永不过期(默认30分钟)expireTime:30# 同步session到数据库的周期(默认1分钟)dbSyncPeriod:1# 相隔多久检查一次session的有效性,默认就是10分钟validationInterval:10# 同一个用户最大会话数,比如2的意思是同一个账号允许最多同时两个人登录(默认-1不限制)maxSession:-1# 踢出之前登录的/之后登录的用户,默认踢出之前登录的用户kickoutAfter:falserememberMe:# 是否开启记住我enabled:true
在
src/main/java/com/ruoyi/framework/config/ShiroConfig.java
类中使用
@Value
注解来注入值
复习:
- 当类中使用了
@Value
和@Bean
注解,需要加@Configuration
注解。@Configuration
用于标识一个类作为应用程序上下文的配置类
4.4、自定义资源映射
在
src/main/java/com/ruoyi/framework/config/ResourcesConfig.java
是静态资源映射的路径:
- 默认的
/
是转发跳转到首页 - 对于上传的文件其实是上传到本地的
D:/ruoyi/uploadPath
路径下
怎么验证呢,我们登录若依,并修改自己的头像,修改好的头像会上传到本机
D:/ruoyi/uploadPath
目录下。
5、Shiro相关
5.1、登录的实现
- 在
src/main/resources/templates/login.html
是我们的登录的 html,在src/main/resources/static/ruoyi/login.js
是我们登录的 js,这里看一下登录表单的验证
表单验证方法为
function validateRule()
方法,若未填写账户密码会提示文字 message。
- 当表单验证成功后会进入
$.validator.setDefaults()
方法,会调用login()
方法进行登录
- 我们看一下后端的登录接口,在
src/main/java/com/ruoyi/web/controller/system/SysLoginController.java
里面的 Post 方法登录:
- 我们自定义的Realm是在
src/main/java/com/ruoyi/framework/shiro/realm/UserRealm.java
中,调用了我们自己的loginService.login()
方法
src/main/java/com/ruoyi/framework/shiro/service/SysLoginService.java
登录逻辑里面做了用户名和密码的校验
5.2、验证码实现
src/main/java/com/ruoyi/framework/config/CaptchaConfig.java
里面可以设置验证码的设置
- 在
application.yml
里面有验证码的开关和验证码的类型
5.3、记住我实现
- 在
login.html
里面有 input 标签,对应login.js
里面会传递一个rememberMe
布尔值
- 在
SysLoginController.java
里面会接收前端传递的布尔值,进而在ShiroConfig
里面进行放置到Cookie中,最终通过securityManager()
方法将这些信息注入到安全管理器里面。
- 如图,在yml中配置cookie的属性
- 将
rememberMeManager
注入到安全管理器securityManager
- 最重要的是ShiroConfig的Shiro过滤器配置,这里表示对于
user,kickout,onlineSession,syncOnlineSession
这些路径点击记住我之后就可以被访问。
5.4、权限控制实现
- 在角色管理可以控制对应角色的菜单权限
在前端代码层面是使用
shiro:hasPermission="xxx:xxx:xxx"
来控制按钮的权限,当然我们也可以加
shiro:hasRole="admin"
这样的话这个按钮就只能由权限字符为 admin 的用户操作了。
当然关于角色权限字符串可以在
sys_role
表中查看:
对于权限标识可以在
sys_menu
中查看:
- 在后端也有相应的权限控制
@RequiresPermissions("xxx.xxx.xxx")
[!note]
为什么前后端都要设置权限呢?
- 前端权限是为了展示出操作按钮,恶意用户总有可能尝试绕过前端直接对后端接口发起请求。 因此,后端必须对每一个请求进行权限检查,确保只有拥有足够权限的用户才能执行相关操作。
5.5、退出
Shiro自身带有退出的过滤器,所以若依是重写了过滤器的方法。
同时在
ShiroConfig
里面指明请求
/logout
路径的过滤器为
logout
,过滤器
logout
指向的就是若依自己写
logoutFilter
,在这个退出过滤器里面重写了方法。
版权归原作者 生命是有光的 所有, 如有侵权,请联系我们删除。