feature(Redis监控): 添加缓存列表

This commit is contained in:
luowenfeng 2022-07-07 12:38:06 +08:00
parent 15ee097a8b
commit 40954e4e46
4 changed files with 259 additions and 17 deletions

View File

@ -5,6 +5,7 @@ import cn.iocoder.yudao.framework.redis.core.RedisKeyDefine;
import cn.iocoder.yudao.framework.redis.core.RedisKeyRegistry; import cn.iocoder.yudao.framework.redis.core.RedisKeyRegistry;
import cn.iocoder.yudao.module.infra.controller.admin.redis.vo.RedisKeyRespVO; import cn.iocoder.yudao.module.infra.controller.admin.redis.vo.RedisKeyRespVO;
import cn.iocoder.yudao.module.infra.controller.admin.redis.vo.RedisMonitorRespVO; import cn.iocoder.yudao.module.infra.controller.admin.redis.vo.RedisMonitorRespVO;
import cn.iocoder.yudao.module.infra.controller.admin.redis.vo.RedisValuesRespVO;
import cn.iocoder.yudao.module.infra.convert.redis.RedisConvert; import cn.iocoder.yudao.module.infra.convert.redis.RedisConvert;
import io.swagger.annotations.Api; import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation; import io.swagger.annotations.ApiOperation;
@ -12,13 +13,10 @@ import org.springframework.data.redis.connection.RedisServerCommands;
import org.springframework.data.redis.core.RedisCallback; import org.springframework.data.redis.core.RedisCallback;
import org.springframework.data.redis.core.StringRedisTemplate; import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.security.access.prepost.PreAuthorize; import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.*;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.annotation.Resource; import javax.annotation.Resource;
import java.util.List; import java.util.*;
import java.util.Properties;
import static cn.iocoder.yudao.framework.common.pojo.CommonResult.success; import static cn.iocoder.yudao.framework.common.pojo.CommonResult.success;
@ -52,4 +50,42 @@ public class RedisController {
return success(RedisConvert.INSTANCE.convertList(keyDefines)); return success(RedisConvert.INSTANCE.convertList(keyDefines));
} }
@GetMapping("/get-key/{keyDefine}")
@ApiOperation("获得 Redis Key")
// @PreAuthorize("@ss.hasPermission('infra:redis:get-key-list')")
public CommonResult<Set<String>> getKeyDefineKeys(@PathVariable("keyDefine") String keyDefine) {
Set<String> Keys = stringRedisTemplate.keys(keyDefine + "*");
return success(Keys);
}
@DeleteMapping("/clear-key/{keyDefine}")
@ApiOperation("获得 Redis Key")
// @PreAuthorize("@ss.hasPermission('infra:redis:get-key-list')")
public CommonResult<Boolean> clearKeyDefineKeys(@PathVariable("keyDefine") String keyDefine) {
stringRedisTemplate.delete(Objects.requireNonNull(stringRedisTemplate.keys(keyDefine + "*")));
return success(Boolean.TRUE);
}
// @PreAuthorize("@ss.hasPermi('monitor:cache:list')")
@GetMapping("/get-key/{cacheName}/{cacheKey}")
public CommonResult<RedisValuesRespVO> getKeyValue(@PathVariable("cacheName") String cacheName, @PathVariable("cacheKey") String cacheKey) {
String cacheValue = stringRedisTemplate.opsForValue().get(cacheKey);
return success(new RedisValuesRespVO(cacheName, cacheKey, cacheValue));
}
// @PreAuthorize("@ss.hasPermi('monitor:cache:list')")
@DeleteMapping("/clearCacheKey/{cacheKey}")
public CommonResult<Boolean> clearCacheKey(@PathVariable String cacheKey) {
stringRedisTemplate.delete(cacheKey);
return success(Boolean.TRUE);
}
// @PreAuthorize("@ss.hasPermi('monitor:cache:list')")
@DeleteMapping("/clearCacheAll")
public CommonResult<Boolean> clearCacheAll() {
Collection<String> cacheKeys = stringRedisTemplate.keys("*");
stringRedisTemplate.delete(cacheKeys);
return success(Boolean.TRUE);
}
} }

View File

@ -0,0 +1,30 @@
package cn.iocoder.yudao.module.infra.controller.admin.redis.vo;
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.Builder;
import lombok.Data;
import org.apache.commons.lang3.StringUtils;
@ApiModel("管理后台 - Redis Key Value onse VO")
@Data
@Builder
public class RedisValuesRespVO {
@ApiModelProperty(value = "oauth2_access_token:%s", required = true, example = "String")
private String keyTemplate;
@ApiModelProperty(value = "c5f6990767804a928f4bb96ca249febf", required = true, example = "String")
private String key;
@ApiModelProperty(required = true, example = "String")
private String value;
public RedisValuesRespVO(String keyTemplate, String key, String value){
this.keyTemplate = StringUtils.replace(keyTemplate, ":", "");
this.key = StringUtils.replace(key, keyTemplate, "");
this.value = value;
}
}

View File

@ -8,10 +8,35 @@ export function getCache() {
}) })
} }
// TODO // 获取模块
export function getKeyList() { export function getKeyList() {
return request({ return request({
url: '/infra/redis/get-key-list', url: '/infra/redis/get-key-list',
method: 'get' method: 'get'
}) })
} }
// 获取键名列表
export function getKeyDefineKeys(keyDefine) {
return request({
url: '/infra/redis/get-key/' + keyDefine,
method: 'get'
})
}
// 获取缓存内容
export function getKeyValue(keyDefine, key) {
return request({
url: '/infra/redis/get-key/' + keyDefine + "/" + key,
method: 'get'
})
}
// 根据键名删除缓存
export function clearCacheKey(key) {
return request({
url: '/infra/redis/clearCacheKey/' + key,
method: 'delete'
})
}

View File

@ -70,24 +70,125 @@
v-loading="keyListLoad" v-loading="keyListLoad"
:data="keyList" :data="keyList"
row-key="id" row-key="id"
@row-click="openCacheInfo"
> >
<el-table-column prop="keyTemplate" label="Key 模板" width="200" /> <el-table-column prop="keyTemplate" label="Key 模板" width="200" />
<el-table-column prop="keyType" label="Key 类型" width="100" /> <el-table-column prop="keyType" label="Key 类型" width="100" />
<el-table-column prop="valueType" label="Value 类型" /> <el-table-column prop="valueType" label="Value 类型" />
<el-table-column prop="timeoutType" label="超时时间" width="200"> <el-table-column prop="timeoutType" label="超时时间" width="200">
<template slot-scope="scope"> <template slot-scope="scope">
<dict-tag :type="DICT_TYPE.INFRA_REDIS_TIMEOUT_TYPE" :value="scope.row.timeoutType" /> <dict-tag
<span v-if="scope.row.timeout > 0">({{ scope.row.timeout / 1000 }} )</span> :type="DICT_TYPE.INFRA_REDIS_TIMEOUT_TYPE"
:value="scope.row.timeoutType"
/>
<span v-if="scope.row.timeout > 0"
>({{ scope.row.timeout / 1000 }} )</span
>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column prop="memo" label="备注" /> <el-table-column prop="memo" label="备注" />
</el-table> </el-table>
</div>
<!-- 缓存模块信息框 -->
<el-dialog
:title="keyTemplate + '模块'"
:visible.sync="open"
width="50vw"
append-to-body
>
<el-row :gutter="10">
<el-col :span="10" class="card-box">
<el-card style="height: 70vh">
<div slot="header">
<span>键名列表</span>
<el-button
style="float: right; padding: 3px 0"
type="text"
icon="el-icon-refresh-right"
@click="refreshCacheKeys"
></el-button>
</div>
<el-table
:data="cachekeys"
style="width: 100%"
@row-click="handleCacheValue"
>
<el-table-column
label="序号"
width="60"
type="index"
></el-table-column>
<el-table-column
label="缓存键名"
align="center"
:show-overflow-tooltip="true"
:formatter="keyFormatter"
>
</el-table-column>
<el-table-column
label="操作"
width="60"
align="center"
class-name="small-padding fixed-width"
>
<template slot-scope="scope">
<el-button
size="mini"
type="text"
icon="el-icon-delete"
@click="handleClearCacheKey(scope.row)"
></el-button>
</template>
</el-table-column>
</el-table>
</el-card>
</el-col>
<el-col :span="14">
<el-card :bordered="false" style="height: 70vh">
<div slot="header">
<span>缓存内容</span>
<!-- <el-button
style="float: right; padding: 3px 0"
type="text"
icon="el-icon-refresh-right"
@click="handleClearCacheAll()"
>清理全部</el-button>
-->
</div>
<el-form :model="cacheForm">
<el-row :gutter="32">
<el-col :offset="1" :span="22">
<el-form-item label="缓存名称:" prop="keyTemplate">
<el-input v-model="cacheForm.keyTemplate" :readOnly="true" />
</el-form-item>
</el-col>
<el-col :offset="1" :span="22">
<el-form-item label="缓存键名:" prop="key">
<el-input v-model="cacheForm.key" :readOnly="true" />
</el-form-item>
</el-col>
<el-col :offset="1" :span="22">
<el-form-item label="缓存内容:" prop="value">
<el-input
v-model="cacheForm.value"
type="textarea"
:rows="12"
:readOnly="true"
/>
</el-form-item>
</el-col>
</el-row>
</el-form>
</el-card>
</el-col>
</el-row>
</el-dialog>
</div>
</template> </template>
<script> <script>
import { getCache, getKeyList } from "@/api/infra/redis"; import { getCache, getKeyList, getKeyDefineKeys, getKeyValue, clearCacheKey } from "@/api/infra/redis";
import echarts from "echarts"; import echarts from "echarts";
export default { export default {
@ -103,6 +204,11 @@ export default {
// key // key
keyListLoad: true, keyListLoad: true,
keyList: [], keyList: [],
//
open: false,
keyTemplate: "",
cachekeys: [],
cacheForm: {}
}; };
}, },
created () { created () {
@ -174,10 +280,55 @@ export default {
this.keyListLoad = false; this.keyListLoad = false;
}); });
}, },
// //
openLoading () { openLoading () {
this.$modal.loading("正在加载缓存监控数据,请稍后!"); this.$modal.loading("正在加载缓存监控数据,请稍后!");
}, },
//
openCacheInfo (e) {
this.open = true;
let keyDefine = e.keyTemplate.substring(0, e.keyTemplate.length - 2);
this.keyTemplate = keyDefine;
//
this.handleCacheKeys(keyDefine);
},
/** 键名前缀去除 */
keyFormatter (cacheKey) {
return cacheKey.replace(this.keyTemplate, "");
},
//
handleCacheKeys (keyDefine){
const cacheName = keyDefine !== undefined ? keyDefine : this.keyTemplate;
getKeyDefineKeys(cacheName).then(response => {
this.cachekeys = response.data
this.cacheForm = {}
})
},
//
handleCacheValue (e){
getKeyValue(this.keyTemplate, e).then(response => {
this.cacheForm = response.data
})
},
//
refreshCacheKeys(){
this.$modal.msgSuccess("刷新键名列表成功");
this.handleCacheKeys();
},
//
handleClearCacheKey(key){
clearCacheKey(key).then(response =>{
this.$modal.msgSuccess("清理缓存键名[" + key + "]成功");
this.handleCacheKeys();
})
},
}, },
}; };
</script> </script>