文章目录
- 前言
- 一、修改
-
- 1.修改错误信息
- 2.修改主页页面
- 二、导航栏改造
-
- 1、添加路由
- 2、添加页面
- 解决前后分离会话问题
-
- 1、导入shiro整合redis的jar包
- 2、自定义SessionManager类
- 3、ShiroConfig注入配置文件SessionManager
-
- 3.1、什么是redis?
- 3.2.下载(文件版)
- 3.3.下载(安装版)
- 3.4、注入SessionManager
- 3.5、SessionManager注入安全管理器
- 四、操作日志页面开发
-
- 1、接口请求
- 2、页面编写
- 3、分页
- 4、中文化
- 五、总结
前言
接下来,我们主要剩下前端页面的编写。vue不懂的学生基本上完成了后端的事情,下面可以选择学习,学习一个vue再看,其实也没什么,希望大家努力学习。
一、修改
1.修改错误信息
我们以前写的登录代码现在可能会发现登录中有错误信息的提示。让我们暂时删除提示。 打开views/login/index.vue文件,修改handleLogin()方法,删除以下代码。
this.$notify.error({
title: 登录失败, message: 请输入正确的用户名或密码! }) 完整代码:
handleLogin() {
this.$refs.loginForm.validate(valid => {
if (valid) {
this.loading = true this.$store.dispatch('Login', this.loginForm).then(() => {
this.loading = false this.$router.push({
path: this.redirect || '/' }) }).catch(() => {
this.loading = false
})
} else {
console.log('error submit!!')
return false
}
})
}
2、主页页面修改
我们修改完,然后运行项目打开页面,会看到如下页面。
我们可以看一下vue-element-admin框架的页面,首页为控制台,我们可以放一些大的标题、图表等信息,美化一下我们的后台网站。 打开src/router/index.js文件,然后我们配置一下路由,将以下的代码注释掉。
{
path: '/',
component: Layout,
redirect: '/dashboard',
name: 'Dashboard',
hidden: true,
children: [{
path: 'dashboard',
component: () => import('@/views/dashboard/index')
}]
},
然后我们自己添加dashboard路由,其中icon的图表可以从https://www.iconfont.cn/进行下载,或者我是下载的vue-element-admin的src/icons/svg中找的。
{
path: '/',
component: Layout,
redirect: '/dashboard',
name: 'Dashboard',
children: [{
path: 'dashboard',
component: () => import('@/views/dashboard/index'),
meta: {
title: 'Dashboard', icon: 'dashboard' }
}]
},
然后我们再打开项目,就会看到我们一进来就是项目的控制台首页了。 这时我们会注意到左侧的导航栏都是英文,我们如何改成自己的导航的文字呢,也是在src/router/index.js这个文件中,我们会看到meta属性,然后修改title的名称即可。
二、导航栏改造
现在我们开始改造一下左侧的导航栏,我们将原来的导航栏都删除掉,只留下dashboard控制台。 如何删除呢?我们先把src/router/index.js里面的删除掉,然后再去views里面找到对应的文件删除掉,如果你不会,可以去https://gitee.com/xyhwh/personal_vue下载。 搭建一个新的页面导航要做以下操作。
1、添加路由
首先我们还是打开src/router/index.js文件,然后我们新添加一个导航的路由。其中icon的图标可以在https://www.iconfont.cn/下载,格式为svg。下载完放到src/icons/svg文件中。
{
path: '/operation',
component: Layout,
redirect: '/operation/list',
name: 'Operation',
meta: {
title: '日志中心', icon: 'operationlog' },
children: [
{
path: 'list',
name: 'OperationList',
component: () => import('@/views/operation/index'),
meta: {
title: '操作日志', icon: '' }
},
{
path: 'loginlog',
name: 'LoginLogList',
component: () => import('@/views/operation/loginlog'),
meta: {
title: '登录日志', icon: '' }
}
]
},
这里我要说一下,路由和侧边栏是组织起一个后台应用的关键骨架。这个框架项目的侧边栏和路由是绑定在一起的,所以你只有在 @/router/index.js 下面配置对应的路由,侧边栏就能动态的生成了。大大减轻了手动重复编辑侧边栏的工作量。当然这样就需要在配置路由的时候遵循一些约定的规则。 例如:
{
path: '/permission',
component: Layout,
redirect: '/permission/index', //重定向地址,在面包屑中点击会重定向去的地址
hidden: true, // 不在侧边栏显示
alwaysShow: true, //一直显示根路由
meta: {
roles: ['admin','editor'] }, //你可以在根路由设置权限,这样它下面所有的子路由都继承了这个权限
children: [{
path: 'index',
component: ()=>import('permission/index'),
name: 'permission',
meta: {
title: 'permission',
icon: 'lock', //图标
roles: ['admin','editor'], //或者你可以给每一个子路由设置自己的权限
noCache: true // 不会被 <keep-alive> 缓存
}
}]
}
前端这个东西要多自己动手看效果,不行就修改,不要记忆,大脑是用来思考的,不是用来记忆的。
2、添加页面
在rcs/views文件下,我们新建一个operation文件夹,然后再里面新建index.vue和loginlog.vue两个文件,我们先创建一些模板。
<template>
<div class="operationlog">
<q>这是操作日志页面</q>
</div>
</template>
<script>
export default {
name: 'Operationlog',
computed: {
}
}
</script>
<style rel="stylesheet/scss" lang="scss" scoped>
</style>
<template>
<div class="operationlog">
<q>这是登录日志页面</q>
</div>
</template>
<script>
export default {
name: 'Operationlog',
computed: {
}
}
</script>
<style rel="stylesheet/scss" lang="scss" scoped>
</style>
然后我们再运行项目,打开浏览器看一下。然后我们可以在右侧开发我们的功能了。 看下我们的后台系统,是不是看着也挺好看的,还是开始的目标,我们知道开发的流程和具体的操作即可,有些安全性的东西和页面美化就不太多的描述,作为毕设系统和初学的项目完全足够了。
三、前后端分离会话问题解决
我再测试的时候发现一个后台的问题,就是我们登录的时候拿到的Cookie没有进行后端的会话存储,导致我们页面请求出现302的问题,前后端的分离问题要考虑会话存储的问题,所以我们现在先解决前后端分离项目中使用shiro的会话问题。要不然页面请求的接口会出现问题。我当时也没考虑页面的问题,只用了postman测了一下。 我这里选择的是redis来存储会话信息 。
1、导入shiro整合redis的jar包
在我们项目的pom.xml中添加如下:
<dependency>
<groupId>org.crazycake</groupId>
<artifactId>shiro-redis</artifactId>
<version>2.4.2.1-RELEASE</version>
</dependency>
2、自定义SessionManager类
我们在项目的shiro包中新建一个MySessionManager.java类,然后这个类继承DefaultWebSessionManager类。
package com.blog.personalblog.shiro;
import org.apache.shiro.web.servlet.ShiroHttpServletRequest;
import org.apache.shiro.web.session.mgt.DefaultWebSessionManager;
import org.apache.shiro.web.util.WebUtils;
import org.springframework.util.StringUtils;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import java.io.Serializable;
/** * @author: SuperMan * @create: 2022-04-14 **/
public class MySessionManager extends DefaultWebSessionManager {
private static final String AUTHORIZATION = "Authorization";
private static final String REFERENCED_SESSION_ID_SOURCE = "Stateless request";
public MySessionManager() {
super();
}
@Override
protected Serializable getSessionId(ServletRequest request, ServletResponse response) {
String id = WebUtils.toHttp(request).getHeader(AUTHORIZATION);
//如果请求头中有 Authorization 则其值为sessionId
if (!StringUtils.isEmpty(id)) {
request.setAttribute(ShiroHttpServletRequest.REFERENCED_SESSION_ID_SOURCE, REFERENCED_SESSION_ID_SOURCE);
request.setAttribute(ShiroHttpServletRequest.REFERENCED_SESSION_ID, id);
request.setAttribute(ShiroHttpServletRequest.REFERENCED_SESSION_ID_IS_VALID, Boolean.TRUE);
return id;
} else {
//否则按默认规则从cookie取sessionId
return super.getSessionId(request, response);
}
}
}
这里大家可能会问,这个类是干嘛的,什么是SessionManager? **SessionManager:**会话管理,即用户登录后就是一次会话,在没有退出之前,它的所有信息都在会话中;会话可以是普通 JavaSE 环境的,也可以是如 Web 环境的,负责管理shiro自己封装的session的生命周期。
1.可以为任意应用提供session支持,不依赖于底层容器 2.简单扩容session管理容器,可以实现任何数据源(redis,ehcache)来管理session,而不必担心jvm内存溢出。 还有想学习更多的可以看下我找的这一篇文章:shiro详解
3、ShiroConfig配置文件注入SessionManager
写到这的时候,我们就需要下载一下redis的软件了,有很多不知道redis是干嘛的,接下来先说一下redis,这也是以后企业经常用到的,大家尽可能的多去了解和使用。
3.1、什么是redis?
Redis(Remote Dictionary Server ),即远程字典服务,是一个开源的使用ANSI C语言编写、支持网络、可基于内存亦可持久化的日志型、Key-Value数据库,并提供多种语言的API。 Redis支持主从同步、是一个高性能的key-value数据库。 redis官网地址:https://redis.io/ 学习教程:https://www.runoob.com/redis/redis-tutorial.html
3.2、下载(文件版)
打开官网即可看到这个download,直接下载即可。 如果想和我的一样,,然后回复:教程,然后关于这个项目的所有软件和资料都会有。我们下载完之后,把压缩包解压后不需要安装,直接点击文件夹里面的redis-server.exe启动即可。 启动效果如下,出现这种情况就启动成功了。 这个缺点是每次启动项目前都要去启动一下redis服务,有点麻烦,我们可以找到redis的安装包,设置成开机自启即可。
3.3、下载(安装版)
下载地址:https://github.com/tporadowski/redis/releases 下载下来之后,直接安装即可。如果下载不下来就去我公众号里下载,上边有说下载方式。 这里有几个常用的redis配置,大家学习一下。 其他的东西大家可以去菜鸟教程上边去学习,网上资料还是很多的,这个不是我们学习的重点。
3.4、注入SessionManager
打开·application.yml·,然后添加如下代码,注意是spring配置下的,redis的密码我的为空,你可以看你安装时是不是设置了密码,如果是解压的基本上都是空。
redis:
host: 127.0.0.1
port: 6379
timeout: 300
password:
data:
redis:
repositories:
enabled: false
然后我们在就在ShiroConfig类中引入这些配置。这里用到了一个注解@Value
@Value("${spring.redis.host}")
private String host;
@Value("${spring.redis..port}")
private int port;
@Value("${spring.redis.timeout}")
private int timeout;
@Value("${spring.redis.password}")
private String password;
/** * 自定义sessionManager,用户的唯一标识,即Token或Authorization的认证 * @return */
@Bean
public SessionManager sessionManager() {
//创建一个上面自定的SessionManager
MySessionManager mySessionManager = new MySessionManager();
mySessionManager.setSessionDAO(redisSessionDAO());
return mySessionManager;
}
在这个方法里面,就使用了我们配置的redis的信息,以后需要改动的时候直接改配置文件即可,也是规范化开发。
/** * 配置shiro redisManager,使用的是shiro-redis开源插件 * * @return */
public RedisManager redisManager() {
RedisManager redisManager = new RedisManager();
redisManager.setHost(host);
redisManager.setPort(port);
// 配置缓存过期时间
redisManager.setExpire(1800);
redisManager.setTimeout(timeout);
redisManager.setPassword(password);
return redisManager;
}
/** * cacheManager 缓存 redis实现 * @return */
@Bean
public RedisCacheManager cacheManager() {
RedisCacheManager redisCacheManager = new RedisCacheManager();
redisCacheManager.setRedisManager(redisManager());
return redisCacheManager;
}
/** * RedisSessionDAO shiro sessionDao层的实现 通过redis * @return */
@Bean
public RedisSessionDAO redisSessionDAO() {
RedisSessionDAO redisSessionDAO = new RedisSessionDAO();
// 自定义session管理 使用redis
redisSessionDAO.setRedisManager(redisManager());
return redisSessionDAO;
}
3.5、SessionManager注入安全管理器
这个方法我们之前已经写过了,只是没有配置redis这个,现在再加入这个。
@Bean
public SecurityManager securityManager() {
DefaultWebSecurityManager securityManager = new DefaultWebSecurityManager();
//一定要通过上面的定义来加载自定义realm,否则在自定义realm中无法注入service层
securityManager.setRealm(myShiroRealm());
// 自定义session管理 使用redis
securityManager.setSessionManager(sessionManager());
// 自定义缓存实现 使用redis
securityManager.setCacheManager(cacheManager());
return securityManager;
}
一切都配置完成之后,我们再修改一下我们之前的配置过滤规则shirFilter这个方法。这里只需要去掉info接口即可。将这个注释掉即可
filterChainDefinitionMap.put("/user/info", "anon");
然后我们再去登录的接口,打开UserController.java,找到info接口,我们这里可以获取到登录的用户了,我们再修改一下。
/** * 登录info信息 * @return */
@GetMapping("/info")
public JsonResult<Object> info(){
String username = (String) SecurityUtils.getSubject().getPrincipal();
User user = userService.getUserByUserName(username);
Map<String, Object> ret = new HashMap<>(3);
ret.put("roles", "[admin]");
ret.put("name", user.getUserName());
ret.put("avatar","https://wpimg.wallstcn.com/f778738c-e4f8-4870-b634-56703b4acafe.gif");
return JsonResult.success(ret);
}
好啦,到现在我们的改造完了,下面去打开前端项目,运行,进行登录一下。看到登录进来name为admin这个是真实的用户,你可以再换一个用户,看看是否会变。 到这里,后端的问题已经解决,下面我们去写页面。
四、操作日志页面开发
在开发的过程中,这个前端框架使用了eslint校验代码,虽然是限制我们规范化写代码,但是特别的烦,动不动就报红色,所以我们先不用这个了,我设置一下就不让它检验src目录下的文件了。打开项目的这个文件,然后加入/src保存即可。 现在开始写页面,我们先加一下操作日志的接口,也就是我们后端写的Controller的api接口,首先打开src/api,然后创建一个operation.js文件,然后写入我们的api接口,和原来的login登录一样,可以参考登录的api。
import request from '@/utils/request'
export function fetchLogList(query) {
return request({
url: '/log/operationLog/list',
method: 'post',
data: query
})
}
1、接口请求
首先打开views/operation/index.vue文件进行编写页面。 我们先写接口,我们首先在script里引入接口
import {
fetchLogList } from '@/api/operation'
然后写获取接口里面的数据,在methods方法里面写。还记得我们封装的返回类,返回的参数是什么嘛,不记得可以去看后端的JsonResult类,我们将数据都封装到了result里面了,我们前端只要去result里面去拿数据即可。数据的请求为JSON格式。
getList() {
this.listLoading = true
var body = this.listQuery;
fetchLogList({
body}).then(response => {
this.list = response.data.result
this.count = response.data.totalSize
this.listLoading = false
})
},
由于我们这里使用的是JSON格式请求,我们的后端也是接收的JSON,所以我们在传递参数的时候,以现在后端的写法是接收不到参数的(你可以自己实验一下,后端的代码可以先不改动,看看参数是否可以接收到),我们现在先去后端改一下Controller接口的接收参数。(注意其余的接口只要是前端传递参数都要按照以下写)。
我们先写一个公共的类,把接口的接收的参数外边再包一层,和controller包同级下新建一个common包,然后里面添加一个PageRequestApi.java类,用于封装请求的参数,也就是参数外边再包一层大括号。
package com.blog.personalblog.common; import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import javax.validation.Valid; /** * @author: SuperMan * @create: 2022-04-26 **/ @JsonIgnoreProperties( ignoreUnknown = true ) public class PageRequestApi<T>