feature: tag功能.

This commit is contained in:
lijiahang
2023-09-05 17:30:37 +08:00
parent 8035396995
commit db9afe3866
25 changed files with 741 additions and 30 deletions

View File

@@ -69,17 +69,17 @@ public class FavoriteApiImpl implements FavoriteApi {
request.setUserId(userId);
request.setType(typeName);
cacheRelIdList = favoriteService.getFavoriteRelIdList(request);
// 设置 -1 到缓存防止穿透
// 添加默认值 防止穿透
if (cacheRelIdList.isEmpty()) {
cacheRelIdList.add(Const.L_N_1);
cacheRelIdList.add(Const.NONE_ID);
}
// 设置缓存
RedisUtils.listPushAll(key, cacheRelIdList, String::valueOf);
// 设置过期时间
RedisUtils.setExpire(key, FavoriteCacheKeyDefine.FAVORITE);
}
// 尝试删除防止穿透的 key
cacheRelIdList.remove(Const.L_N_1);
// 删除防止穿透的 key
cacheRelIdList.remove(Const.NONE_ID);
return CompletableFuture.completedFuture(cacheRelIdList);
}

View File

@@ -0,0 +1,23 @@
### 创建标签枚举
POST {{baseUrl}}/infra/tag/create
Content-Type: application/json
Authorization: {{token}}
{
"name": "TAG1",
"type": "HOST"
}
### 查询标签枚举
GET {{baseUrl}}/infra/tag/list?type=HOST
Content-Type: application/json
Authorization: {{token}}
### 通过 id 删除标签枚举
DELETE {{baseUrl}}/infra/tag/delete?id=2
Authorization: {{token}}
###

View File

@@ -0,0 +1,64 @@
package com.orion.ops.module.infra.controller;
import com.orion.ops.framework.common.annotation.IgnoreLog;
import com.orion.ops.framework.common.annotation.RestWrapper;
import com.orion.ops.framework.common.constant.IgnoreLogMode;
import com.orion.ops.module.infra.entity.request.tag.TagCreateRequest;
import com.orion.ops.module.infra.entity.vo.TagVO;
import com.orion.ops.module.infra.service.TagService;
import io.swagger.v3.oas.annotations.Operation;
import io.swagger.v3.oas.annotations.Parameter;
import io.swagger.v3.oas.annotations.tags.Tag;
import lombok.extern.slf4j.Slf4j;
import org.springframework.security.access.prepost.PreAuthorize;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.*;
import javax.annotation.Resource;
import java.util.List;
/**
* 标签枚举 api
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-5 11:58
*/
@Tag(name = "infra - 标签枚举服务")
@Slf4j
@Validated
@RestWrapper
@RestController
@RequestMapping("/infra/tag")
@SuppressWarnings({"ELValidationInJSP", "SpringElInspection"})
public class TagController {
@Resource
private TagService tagService;
@PostMapping("/create")
@Operation(summary = "创建标签枚举")
@PreAuthorize("@ss.hasPermission('infra:tag:create')")
public Long createTag(@Validated @RequestBody TagCreateRequest request) {
return tagService.createTag(request);
}
@IgnoreLog(IgnoreLogMode.RET)
@GetMapping("/list")
@Operation(summary = "查询标签枚举")
@Parameter(name = "type", description = "type", required = true)
@PreAuthorize("@ss.hasPermission('infra:tag:query')")
public List<TagVO> getTagListAll(@RequestParam("type") String type) {
return tagService.getTagList(type);
}
@DeleteMapping("/delete")
@Operation(summary = "通过 id 删除标签枚举")
@Parameter(name = "id", description = "id", required = true)
@PreAuthorize("@ss.hasPermission('infra:tag:delete')")
public Integer deleteTag(@RequestParam("id") Long id) {
return tagService.deleteTagById(id);
}
}

View File

@@ -0,0 +1,37 @@
package com.orion.ops.module.infra.convert;
import com.orion.ops.module.infra.entity.domain.TagDO;
import com.orion.ops.module.infra.entity.dto.TagCacheDTO;
import com.orion.ops.module.infra.entity.request.tag.TagCreateRequest;
import com.orion.ops.module.infra.entity.request.tag.TagQueryRequest;
import com.orion.ops.module.infra.entity.vo.TagVO;
import org.mapstruct.Mapper;
import org.mapstruct.factory.Mappers;
import java.util.List;
/**
* 标签枚举 内部对象转换器
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-5 11:58
*/
@Mapper
public interface TagConvert {
TagConvert MAPPER = Mappers.getMapper(TagConvert.class);
TagDO to(TagCreateRequest request);
TagDO to(TagQueryRequest request);
TagVO to(TagDO domain);
TagCacheDTO toCache(TagDO domain);
List<TagVO> to(List<TagDO> list);
List<TagVO> toList(List<TagCacheDTO> cache);
}

View File

@@ -0,0 +1,31 @@
package com.orion.ops.module.infra.dao;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.orion.ops.framework.mybatis.core.mapper.IMapper;
import com.orion.ops.module.infra.entity.domain.TagDO;
import org.apache.ibatis.annotations.Mapper;
/**
* 标签枚举 Mapper 接口
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-5 11:58
*/
@Mapper
public interface TagDAO extends IMapper<TagDO> {
/**
* 获取查询条件
*
* @param entity entity
* @return 查询条件
*/
default LambdaQueryWrapper<TagDO> queryCondition(TagDO entity) {
return this.wrapper()
.eq(TagDO::getId, entity.getId())
.eq(TagDO::getName, entity.getName())
.eq(TagDO::getType, entity.getType());
}
}

View File

@@ -0,0 +1,25 @@
package com.orion.ops.module.infra.define;
import com.orion.lang.define.cache.CacheKeyBuilder;
import com.orion.lang.define.cache.CacheKeyDefine;
import com.orion.ops.module.infra.entity.dto.TagCacheDTO;
import java.util.concurrent.TimeUnit;
/**
* tag 服务缓存 key
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023/9/5 15:36
*/
public interface TagCacheKeyDefine {
CacheKeyDefine TAG_NAME = new CacheKeyBuilder()
.key("tag:{}")
.desc("tag名称 ${type}")
.type(TagCacheDTO.class)
.timeout(3, TimeUnit.DAYS)
.build();
}

View File

@@ -0,0 +1,41 @@
package com.orion.ops.module.infra.entity.domain;
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName;
import com.orion.ops.framework.mybatis.core.domain.BaseDO;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.*;
/**
* 标签枚举 实体对象
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-5 11:58
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@EqualsAndHashCode(callSuper = true)
@TableName(value = "tag", autoResultMap = true)
@Schema(name = "TagDO", description = "标签枚举 实体对象")
public class TagDO extends BaseDO {
private static final long serialVersionUID = 1L;
@Schema(description = "id")
@TableId(value = "id", type = IdType.AUTO)
private Long id;
@Schema(description = "标签名称")
@TableField("name")
private String name;
@Schema(description = "标签类型")
@TableField("type")
private String type;
}

View File

@@ -0,0 +1,31 @@
package com.orion.ops.module.infra.entity.dto;
import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.annotation.TableId;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.*;
/**
* @author Jiahang Li
* @version 1.0.0
* @since 2023/9/5 15:37
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@EqualsAndHashCode(onlyExplicitlyIncluded = true)
@Schema(name = "TagCacheDTO", description = "菜单 缓存业务对象")
public class TagCacheDTO {
@EqualsAndHashCode.Include
@Schema(description = "id")
@TableId(value = "id", type = IdType.AUTO)
private Long id;
@Schema(description = "标签名称")
@TableField("name")
private String name;
}

View File

@@ -0,0 +1,37 @@
package com.orion.ops.module.infra.entity.request.tag;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;
import javax.validation.constraints.NotBlank;
import javax.validation.constraints.Size;
import java.io.Serializable;
/**
* 标签枚举 创建请求对象
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-5 11:58
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@Schema(name = "TagCreateRequest", description = "标签枚举 创建请求对象")
public class TagCreateRequest implements Serializable {
@NotBlank
@Size(max = 32)
@Schema(description = "标签名称")
private String name;
@NotBlank
@Size(max = 12)
@Schema(description = "标签类型")
private String type;
}

View File

@@ -0,0 +1,28 @@
package com.orion.ops.module.infra.entity.request.tag;
import com.orion.ops.framework.common.entity.PageRequest;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.*;
import javax.validation.constraints.Size;
/**
* 标签枚举 查询请求对象
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-5 11:58
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@EqualsAndHashCode(callSuper = true)
@Schema(name = "TagQueryRequest", description = "标签枚举 查询请求对象")
public class TagQueryRequest extends PageRequest {
@Size(max = 12)
@Schema(description = "标签类型")
private String type;
}

View File

@@ -0,0 +1,42 @@
package com.orion.ops.module.infra.entity.request.tag;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;
import javax.validation.constraints.NotBlank;
import javax.validation.constraints.NotNull;
import javax.validation.constraints.Size;
import java.io.Serializable;
/**
* 标签枚举 更新请求对象
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-5 11:58
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@Schema(name = "TagUpdateRequest", description = "标签枚举 更新请求对象")
public class TagUpdateRequest implements Serializable {
@NotNull
@Schema(description = "id")
private Long id;
@NotBlank
@Size(max = 32)
@Schema(description = "标签名称")
private String name;
@NotBlank
@Size(max = 12)
@Schema(description = "标签类型")
private String type;
}

View File

@@ -0,0 +1,33 @@
package com.orion.ops.module.infra.entity.vo;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.io.Serializable;
/**
* 标签枚举 视图响应对象
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-5 11:58
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@Schema(name = "TagVO", description = "标签枚举 视图响应对象")
public class TagVO implements Serializable {
private static final long serialVersionUID = 1L;
@Schema(description = "id")
private Long id;
@Schema(description = "标签名称")
private String name;
}

View File

@@ -0,0 +1,49 @@
package com.orion.ops.module.infra.service;
import com.orion.ops.module.infra.entity.request.tag.TagCreateRequest;
import com.orion.ops.module.infra.entity.vo.TagVO;
import java.util.List;
/**
* 标签枚举 服务类
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-5 11:58
*/
public interface TagService {
/**
* 创建标签枚举
*
* @param request request
* @return id
*/
Long createTag(TagCreateRequest request);
/**
* 查询标签枚举
*
* @param type type
* @return rows
*/
List<TagVO> getTagList(String type);
/**
* 通过 id 删除标签枚举
*
* @param id id
* @return effect
*/
Integer deleteTagById(Long id);
/**
* 通过 id 删除标签枚举
*
* @param idList idList
* @return effect
*/
Integer deleteTagByIdList(List<Long> idList);
}

View File

@@ -0,0 +1,118 @@
package com.orion.ops.module.infra.service.impl;
import com.alibaba.fastjson.JSON;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.orion.ops.framework.common.constant.Const;
import com.orion.ops.framework.mybatis.core.query.Conditions;
import com.orion.ops.framework.redis.core.utils.RedisUtils;
import com.orion.ops.module.infra.convert.TagConvert;
import com.orion.ops.module.infra.dao.TagDAO;
import com.orion.ops.module.infra.define.TagCacheKeyDefine;
import com.orion.ops.module.infra.entity.domain.TagDO;
import com.orion.ops.module.infra.entity.dto.TagCacheDTO;
import com.orion.ops.module.infra.entity.request.tag.TagCreateRequest;
import com.orion.ops.module.infra.entity.vo.TagVO;
import com.orion.ops.module.infra.service.TagService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import javax.annotation.Resource;
import java.util.List;
/**
* 标签枚举 服务实现类
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-5 11:58
*/
@Slf4j
@Service
public class TagServiceImpl implements TagService {
@Resource
private TagDAO tagDAO;
@Override
public Long createTag(TagCreateRequest request) {
// 转换
TagDO record = TagConvert.MAPPER.to(request);
record.setId(null);
// 查询 tag 是否存在
String type = record.getType();
LambdaQueryWrapper<TagDO> wrapper = tagDAO.wrapper()
.eq(TagDO::getName, record.getName())
.eq(TagDO::getType, type);
TagDO checkTag = tagDAO.of(wrapper)
.only()
.get();
if (checkTag != null) {
return checkTag.getId();
}
// 插入
int effect = tagDAO.insert(record);
log.info("TagService-createTag effect: {}, record: {}", effect, JSON.toJSONString(record));
// 设置缓存
String cacheKey = TagCacheKeyDefine.TAG_NAME.format(type);
TagCacheDTO cache = TagConvert.MAPPER.toCache(record);
RedisUtils.listPushJson(cacheKey, cache);
RedisUtils.setExpire(cacheKey, TagCacheKeyDefine.TAG_NAME);
return record.getId();
}
@Override
public List<TagVO> getTagList(String type) {
// 查询缓存
String cacheKey = TagCacheKeyDefine.TAG_NAME.format(type);
List<TagCacheDTO> cacheValues = RedisUtils.listRangeJson(cacheKey, TagCacheKeyDefine.TAG_NAME);
if (cacheValues.isEmpty()) {
// 为空则需要查询缓存
LambdaQueryWrapper<TagDO> wrapper = Conditions.eq(TagDO::getType, type);
cacheValues = tagDAO.of(wrapper).list(TagConvert.MAPPER::toCache);
// 添加默认值 防止穿透
if (cacheValues.isEmpty()) {
cacheValues.add(TagCacheDTO.builder().id(Const.NONE_ID).build());
}
// 设置到缓存
RedisUtils.listPushAllJson(cacheKey, cacheValues);
RedisUtils.setExpire(cacheKey, TagCacheKeyDefine.TAG_NAME);
}
// 删除防止穿透的 key
cacheValues.removeIf(s -> Const.NONE_ID.equals(s.getId()));
// 转换
return TagConvert.MAPPER.toList(cacheValues);
}
@Override
public Integer deleteTagById(Long id) {
TagDO tag = tagDAO.selectById(id);
if (tag == null) {
return Const.N_0;
}
// 删除数据库
int effect = tagDAO.deleteById(id);
log.info("TagService-deleteTagById id: {}, effect: {}", id, effect);
// 删除缓存
String cacheKey = TagCacheKeyDefine.TAG_NAME.format(tag.getType());
RedisUtils.listRemoveJson(cacheKey, TagConvert.MAPPER.toCache(tag));
return effect;
}
@Override
public Integer deleteTagByIdList(List<Long> idList) {
List<TagDO> tagList = tagDAO.selectBatchIds(idList);
if (tagList.isEmpty()) {
return Const.N_0;
}
// 删除数据库
int effect = tagDAO.deleteBatchIds(idList);
log.info("TagService-deleteTagByIdList idList: {}, effect: {}", idList, effect);
// 删除缓存
for (TagDO tag : tagList) {
String cacheKey = TagCacheKeyDefine.TAG_NAME.format(tag.getType());
RedisUtils.listRemoveJson(cacheKey, TagConvert.MAPPER.toCache(tag));
}
return effect;
}
}

View File

@@ -0,0 +1,22 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.orion.ops.module.infra.dao.TagDAO">
<!-- 通用查询映射结果 -->
<resultMap id="BaseResultMap" type="com.orion.ops.module.infra.entity.domain.TagDO">
<id column="id" property="id"/>
<result column="create_time" property="createTime"/>
<result column="update_time" property="updateTime"/>
<result column="creator" property="creator"/>
<result column="updater" property="updater"/>
<result column="deleted" property="deleted"/>
<result column="name" property="name"/>
<result column="type" property="type"/>
</resultMap>
<!-- 通用查询结果列 -->
<sql id="Base_Column_List">
id, name, type, create_time, update_time, creator, updater, deleted
</sql>
</mapper>