添加主机 api.

This commit is contained in:
lijiahang
2023-09-13 14:24:16 +08:00
parent 3990090155
commit 973bad39b1
20 changed files with 1204 additions and 0 deletions

View File

@@ -0,0 +1,64 @@
### 创建主机
POST {{baseUrl}}/asset/host/create
Content-Type: application/json
Authorization: {{token}}
{
"name": "",
"code": "",
"address": ""
}
### 通过 id 更新主机
PUT {{baseUrl}}/asset/host/update
Content-Type: application/json
Authorization: {{token}}
{
"id": "",
"name": "",
"code": "",
"address": ""
}
### 通过 id 查询主机
GET {{baseUrl}}/asset/host/get?id=1&extra=true&config=true
Authorization: {{token}}
### 查询主机
POST {{baseUrl}}/asset/host/list-all
Content-Type: application/json
Authorization: {{token}}
{
"id": "",
"name": "",
"code": "",
"address": ""
}
### 分页查询主机
POST {{baseUrl}}/asset/host/query
Content-Type: application/json
Authorization: {{token}}
{
"page": 1,
"limit": 10,
"id": "",
"name": "",
"code": "",
"address": ""
}
### 通过 id 删除主机
DELETE {{baseUrl}}/asset/host/delete?id=1
Authorization: {{token}}
###

View File

@@ -0,0 +1,100 @@
package com.orion.ops.module.asset.controller;
import com.orion.lang.define.wrapper.DataGrid;
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.framework.common.valid.group.Id;
import com.orion.ops.framework.common.valid.group.Page;
import com.orion.ops.module.asset.entity.request.host.HostCreateRequest;
import com.orion.ops.module.asset.entity.request.host.HostQueryRequest;
import com.orion.ops.module.asset.entity.request.host.HostUpdateRequest;
import com.orion.ops.module.asset.entity.vo.HostVO;
import com.orion.ops.module.asset.service.HostService;
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-11 14:16
*/
@Tag(name = "asset - 主机服务")
@Slf4j
@Validated
@RestWrapper
@RestController
@RequestMapping("/asset/host")
@SuppressWarnings({"ELValidationInJSP", "SpringElInspection"})
public class HostController {
@Resource
private HostService hostService;
@PostMapping("/create")
@Operation(summary = "创建主机")
@PreAuthorize("@ss.hasPermission('asset:host:create')")
public Long createHost(@Validated @RequestBody HostCreateRequest request) {
return hostService.createHost(request);
}
@PutMapping("/update")
@Operation(summary = "通过 id 更新主机")
@PreAuthorize("@ss.hasPermission('asset:host:update')")
public Integer updateHost(@Validated(Id.class) @RequestBody HostUpdateRequest request) {
return hostService.updateHostById(request);
}
@IgnoreLog(IgnoreLogMode.RET)
@GetMapping("/get")
@Operation(summary = "通过 id 查询主机")
@Parameter(name = "id", description = "id", required = true)
@Parameter(name = "extra", description = "是否查询额外信息")
@Parameter(name = "config", description = "是否查询配置信息")
@PreAuthorize("@ss.hasPermission('asset:host:query')")
public HostVO getHost(@RequestParam("id") Long id,
@RequestParam(name = "extra", required = false) boolean extra,
@RequestParam(name = "config", required = false) boolean config) {
HostQueryRequest request = new HostQueryRequest();
request.setId(id);
request.setExtra(extra);
request.setConfig(config);
return hostService.getHostById(request);
}
@IgnoreLog(IgnoreLogMode.RET)
@PostMapping("/list-all")
@Operation(summary = "查询主机")
@PreAuthorize("@ss.hasPermission('asset:host:query')")
public List<HostVO> getHostListAll(@Validated @RequestBody HostQueryRequest request) {
return hostService.getHostList(request);
}
@IgnoreLog(IgnoreLogMode.RET)
@PostMapping("/query")
@Operation(summary = "分页查询主机")
@PreAuthorize("@ss.hasPermission('asset:host:query')")
public DataGrid<HostVO> getHostPage(@Validated(Page.class) @RequestBody HostQueryRequest request) {
return hostService.getHostPage(request);
}
@DeleteMapping("/delete")
@Operation(summary = "通过 id 删除主机")
@Parameter(name = "id", description = "id", required = true)
@PreAuthorize("@ss.hasPermission('asset:host:delete')")
public Integer deleteHost(@RequestParam("id") Long id) {
return hostService.deleteHostById(id);
}
}

View File

@@ -0,0 +1,22 @@
package com.orion.ops.module.asset.convert;
import com.orion.ops.module.asset.entity.domain.HostConfigDO;
import com.orion.ops.module.asset.entity.vo.HostConfigVO;
import org.mapstruct.Mapper;
import org.mapstruct.factory.Mappers;
/**
* 主机配置 内部对象转换器
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-11 14:16
*/
@Mapper
public interface HostConfigConvert {
HostConfigConvert MAPPER = Mappers.getMapper(HostConfigConvert.class);
HostConfigVO to(HostConfigDO domain);
}

View File

@@ -0,0 +1,35 @@
package com.orion.ops.module.asset.convert;
import com.orion.ops.module.asset.entity.domain.HostDO;
import com.orion.ops.module.asset.entity.request.host.HostCreateRequest;
import com.orion.ops.module.asset.entity.request.host.HostQueryRequest;
import com.orion.ops.module.asset.entity.request.host.HostUpdateRequest;
import com.orion.ops.module.asset.entity.vo.HostVO;
import org.mapstruct.Mapper;
import org.mapstruct.factory.Mappers;
import java.util.List;
/**
* 主机 内部对象转换器
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-11 14:16
*/
@Mapper
public interface HostConvert {
HostConvert MAPPER = Mappers.getMapper(HostConvert.class);
HostDO to(HostCreateRequest request);
HostDO to(HostUpdateRequest request);
HostDO to(HostQueryRequest request);
HostVO to(HostDO domain);
List<HostVO> to(List<HostDO> list);
}

View File

@@ -0,0 +1,92 @@
package com.orion.ops.module.asset.dao;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.orion.ops.framework.mybatis.core.mapper.IMapper;
import com.orion.ops.module.asset.entity.domain.HostConfigDO;
import org.apache.ibatis.annotations.Mapper;
import java.util.List;
/**
* 主机配置 Mapper 接口
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-11 14:16
*/
@Mapper
public interface HostConfigDAO extends IMapper<HostConfigDO> {
/**
* 通过 hostId 查询主机配置
*
* @param hostId hostId
* @param type type
* @return row
*/
default HostConfigDO getHostConfigByHostId(Long hostId, String type) {
// 条件
LambdaQueryWrapper<HostConfigDO> wrapper = this.lambda()
.eq(HostConfigDO::getHostId, hostId)
.eq(HostConfigDO::getType, type);
// 查询
return this.of(wrapper).getOne();
}
/**
* 通过 hostId 查询主机配置
*
* @param hostId hostId
* @return rows
*/
default List<HostConfigDO> getHostConfigByHostId(Long hostId) {
// 条件
LambdaQueryWrapper<HostConfigDO> wrapper = this.lambda()
.eq(HostConfigDO::getHostId, hostId);
// 查询
return this.of(wrapper).list();
}
/**
* 通过 hostId 批量查询主机配置
*
* @param hostIdList hostIdList
* @return rows
*/
default List<HostConfigDO> getHostConfigByHostIdList(List<Long> hostIdList) {
// 条件
LambdaQueryWrapper<HostConfigDO> wrapper = this.lambda()
.in(HostConfigDO::getHostId, hostIdList);
// 查询
return this.of(wrapper).list();
}
/**
* 通过 hostId 删除主机配置
*
* @param hostId hostId
* @return effect
*/
default Integer deleteByHostId(Long hostId) {
// 条件
LambdaQueryWrapper<HostConfigDO> wrapper = this.lambda()
.eq(HostConfigDO::getHostId, hostId);
// 删除
return this.delete(wrapper);
}
/**
* 通过 hostId 批量删除主机配置
*
* @param hostIdList hostIdList
* @return effect
*/
default Integer deleteByHostIdList(List<Long> hostIdList) {
// 条件
LambdaQueryWrapper<HostConfigDO> wrapper = this.lambda()
.in(HostConfigDO::getHostId, hostIdList);
// 删除
return this.delete(wrapper);
}
}

View File

@@ -0,0 +1,32 @@
package com.orion.ops.module.asset.dao;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.orion.ops.framework.mybatis.core.mapper.IMapper;
import com.orion.ops.module.asset.entity.domain.HostDO;
import org.apache.ibatis.annotations.Mapper;
/**
* 主机 Mapper 接口
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-11 14:16
*/
@Mapper
public interface HostDAO extends IMapper<HostDO> {
/**
* 获取查询条件
*
* @param entity entity
* @return 查询条件
*/
default LambdaQueryWrapper<HostDO> queryCondition(HostDO entity) {
return this.wrapper()
.eq(HostDO::getId, entity.getId())
.eq(HostDO::getName, entity.getName())
.eq(HostDO::getCode, entity.getCode())
.eq(HostDO::getAddress, entity.getAddress());
}
}

View File

@@ -0,0 +1,49 @@
package com.orion.ops.module.asset.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-11 14:16
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@EqualsAndHashCode(callSuper = true)
@TableName(value = "host_config", autoResultMap = true)
@Schema(name = "HostConfigDO", description = "主机配置 实体对象")
public class HostConfigDO extends BaseDO {
private static final long serialVersionUID = 1L;
@Schema(description = "id")
@TableId(value = "id", type = IdType.AUTO)
private Long id;
@Schema(description = "主机id")
@TableField("host_id")
private Long hostId;
@Schema(description = "连接类型")
@TableField("type")
private String type;
@Schema(description = "配置详情")
@TableField("config")
private String config;
@Schema(description = "配置版本号")
@TableField("version")
private String version;
}

View File

@@ -0,0 +1,45 @@
package com.orion.ops.module.asset.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-11 14:16
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@EqualsAndHashCode(callSuper = true)
@TableName(value = "host", autoResultMap = true)
@Schema(name = "HostDO", description = "主机 实体对象")
public class HostDO 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("code")
private String code;
@Schema(description = "主机地址")
@TableField("address")
private String address;
}

View File

@@ -0,0 +1,46 @@
package com.orion.ops.module.asset.entity.request.host;
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;
import java.util.List;
/**
* 主机 创建请求对象
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-11 14:16
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@Schema(name = "HostCreateRequest", description = "主机 创建请求对象")
public class HostCreateRequest implements Serializable {
@NotBlank
@Size(max = 64)
@Schema(description = "主机名称")
private String name;
@NotBlank
@Size(max = 64)
@Schema(description = "主机编码")
private String code;
@NotBlank
@Size(max = 128)
@Schema(description = "主机地址")
private String address;
@Schema(description = "tags")
private List<Long> tags;
}

View File

@@ -0,0 +1,52 @@
package com.orion.ops.module.asset.entity.request.host;
import com.orion.ops.framework.common.entity.PageRequest;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.*;
import javax.validation.constraints.Size;
import java.util.List;
/**
* 主机 查询请求对象
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-11 14:16
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@EqualsAndHashCode(callSuper = true)
@Schema(name = "HostQueryRequest", description = "主机 查询请求对象")
public class HostQueryRequest extends PageRequest {
@Schema(description = "id")
private Long id;
@Size(max = 64)
@Schema(description = "主机名称")
private String name;
@Size(max = 64)
@Schema(description = "主机编码")
private String code;
@Size(max = 128)
@Schema(description = "主机地址")
private String address;
@Schema(description = "是否为收藏")
private Boolean favorite;
@Schema(description = "tag")
private List<Long> tags;
@Schema(description = "是否查询额外信息")
private Boolean extra;
@Schema(description = "是否查询配置信息")
private Boolean config;
}

View File

@@ -0,0 +1,51 @@
package com.orion.ops.module.asset.entity.request.host;
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;
import java.util.List;
/**
* 主机 更新请求对象
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-11 14:16
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@Schema(name = "HostUpdateRequest", description = "主机 更新请求对象")
public class HostUpdateRequest implements Serializable {
@NotNull
@Schema(description = "id")
private Long id;
@NotBlank
@Size(max = 64)
@Schema(description = "主机名称")
private String name;
@NotBlank
@Size(max = 64)
@Schema(description = "主机编码")
private String code;
@NotBlank
@Size(max = 128)
@Schema(description = "主机地址")
private String address;
@Schema(description = "tags")
private List<Long> tags;
}

View File

@@ -0,0 +1,34 @@
package com.orion.ops.module.asset.entity.vo;
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.util.Map;
/**
* 主机配置 视图响应对象
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023/9/11 17:58
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@Schema(name = "HostConfigVO", description = "主机配置 视图响应对象")
public class HostConfigVO {
@Schema(description = "id")
private Long id;
@Schema(description = "version")
private String version;
@Schema(description = "config")
private Map<String, Object> config;
}

View File

@@ -0,0 +1,62 @@
package com.orion.ops.module.asset.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;
import java.util.Date;
import java.util.Map;
/**
* 主机 视图响应对象
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-11 14:16
*/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
@Schema(name = "HostVO", description = "主机 视图响应对象")
public class HostVO implements Serializable {
private static final long serialVersionUID = 1L;
@Schema(description = "id")
private Long id;
@Schema(description = "主机名称")
private String name;
@Schema(description = "主机编码")
private String code;
@Schema(description = "主机地址")
private String address;
@Schema(description = "创建时间")
private Date createTime;
@Schema(description = "修改时间")
private Date updateTime;
@Schema(description = "创建人")
private String creator;
@Schema(description = "修改人")
private String updater;
@Schema(description = "是否收藏")
private Boolean favorite;
@Schema(description = "tags")
private Map<Long, String> tags;
@Schema(description = "configs")
private Map<String, HostConfigVO> configs;
}

View File

@@ -0,0 +1,37 @@
package com.orion.ops.module.asset.enums;
/**
* 主机配置类型枚举
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023/9/11 14:37
*/
public enum HostConfigTypeEnum {
/**
* ssh 配置
*/
SSH,
/**
* sftp 配置
*/
SFTP,
/**
* rdp 配置
*/
RDP,
/**
* 环境变量
*/
ENV,
/**
* 普通配置
*/
CONFIG,
}

View File

@@ -0,0 +1,12 @@
package com.orion.ops.module.asset.service;
/**
* 主机配置 服务类
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-11 14:16
*/
public interface HostConfigService {
}

View File

@@ -0,0 +1,68 @@
package com.orion.ops.module.asset.service;
import com.orion.lang.define.wrapper.DataGrid;
import com.orion.ops.module.asset.entity.request.host.HostCreateRequest;
import com.orion.ops.module.asset.entity.request.host.HostQueryRequest;
import com.orion.ops.module.asset.entity.request.host.HostUpdateRequest;
import com.orion.ops.module.asset.entity.vo.HostVO;
import java.util.List;
/**
* 主机 服务类
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-11 14:16
*/
public interface HostService {
/**
* 创建主机
*
* @param request request
* @return id
*/
Long createHost(HostCreateRequest request);
/**
* 通过 id 更新主机
*
* @param request request
* @return effect
*/
Integer updateHostById(HostUpdateRequest request);
/**
* 通过 id 查询主机
*
* @param request request
* @return row
*/
HostVO getHostById(HostQueryRequest request);
/**
* 查询主机
*
* @param request request
* @return rows
*/
List<HostVO> getHostList(HostQueryRequest request);
/**
* 分页查询主机
*
* @param request request
* @return rows
*/
DataGrid<HostVO> getHostPage(HostQueryRequest request);
/**
* 通过 id 删除主机
*
* @param id id
* @return effect
*/
Integer deleteHostById(Long id);
}

View File

@@ -0,0 +1,24 @@
package com.orion.ops.module.asset.service.impl;
import com.orion.ops.module.asset.dao.HostConfigDAO;
import com.orion.ops.module.asset.service.HostConfigService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import javax.annotation.Resource;
/**
* 主机配置 服务实现类
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-11 14:16
*/
@Slf4j
@Service
public class HostConfigServiceImpl implements HostConfigService {
@Resource
private HostConfigDAO hostConfigDAO;
}

View File

@@ -0,0 +1,332 @@
package com.orion.ops.module.asset.service.impl;
import com.alibaba.fastjson.JSON;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.orion.lang.define.wrapper.DataGrid;
import com.orion.lang.utils.Booleans;
import com.orion.lang.utils.collect.Lists;
import com.orion.ops.framework.common.constant.ErrorMessage;
import com.orion.ops.framework.common.utils.Valid;
import com.orion.ops.framework.security.core.utils.SecurityUtils;
import com.orion.ops.module.asset.convert.HostConfigConvert;
import com.orion.ops.module.asset.convert.HostConvert;
import com.orion.ops.module.asset.dao.HostConfigDAO;
import com.orion.ops.module.asset.dao.HostDAO;
import com.orion.ops.module.asset.entity.domain.HostConfigDO;
import com.orion.ops.module.asset.entity.domain.HostDO;
import com.orion.ops.module.asset.entity.request.host.HostCreateRequest;
import com.orion.ops.module.asset.entity.request.host.HostQueryRequest;
import com.orion.ops.module.asset.entity.request.host.HostUpdateRequest;
import com.orion.ops.module.asset.entity.vo.HostConfigVO;
import com.orion.ops.module.asset.entity.vo.HostVO;
import com.orion.ops.module.asset.service.HostService;
import com.orion.ops.module.infra.api.FavoriteApi;
import com.orion.ops.module.infra.api.TagRelApi;
import com.orion.ops.module.infra.entity.dto.tag.TagDTO;
import com.orion.ops.module.infra.enums.FavoriteTypeEnum;
import com.orion.ops.module.infra.enums.TagTypeEnum;
import lombok.SneakyThrows;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import javax.annotation.Resource;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.Future;
import java.util.stream.Collectors;
/**
* 主机 服务实现类
*
* @author Jiahang Li
* @version 1.0.0
* @since 2023-9-11 14:16
*/
@Slf4j
@Service
public class HostServiceImpl implements HostService {
private static final ThreadLocal<List<Long>> FAVORITE_HOLDER = new ThreadLocal<>();
@Resource
private HostDAO hostDAO;
@Resource
private HostConfigDAO hostConfigDAO;
@Resource
private TagRelApi tagRelApi;
@Resource
private FavoriteApi favoriteApi;
@Override
public Long createHost(HostCreateRequest request) {
log.info("HostService-createHost request: {}", JSON.toJSONString(request));
// 转换
HostDO record = HostConvert.MAPPER.to(request);
// 查询数据是否冲突
this.checkHostNamePresent(record);
this.checkHostCodePresent(record);
// 插入主机
int effect = hostDAO.insert(record);
log.info("HostService-createHost effect: {}", effect);
Long id = record.getId();
// 插入 tag
List<Long> tags = request.getTags();
if (!Lists.isEmpty(tags)) {
tagRelApi.addTagRel(TagTypeEnum.HOST, id, tags);
}
return id;
}
@Override
public Integer updateHostById(HostUpdateRequest request) {
log.info("HostService-updateHostById request: {}", JSON.toJSONString(request));
// 查询
Long id = Valid.notNull(request.getId(), ErrorMessage.ID_MISSING);
HostDO record = hostDAO.selectById(id);
Valid.notNull(record, ErrorMessage.DATA_ABSENT);
// 转换
HostDO updateRecord = HostConvert.MAPPER.to(request);
// 查询数据是否冲突
this.checkHostNamePresent(updateRecord);
this.checkHostCodePresent(updateRecord);
// 更新
int effect = hostDAO.updateById(updateRecord);
log.info("HostService-updateHostById effect: {}", effect);
// 更新 tag
tagRelApi.setTagRel(TagTypeEnum.HOST, id, request.getTags());
return effect;
}
@Override
public HostVO getHostById(HostQueryRequest request) {
// 查询
HostDO record = hostDAO.selectById(request.getId());
Valid.notNull(record, ErrorMessage.DATA_ABSENT);
// 转换
HostVO vo = HostConvert.MAPPER.to(record);
// 查询拓展信息
this.setExtraInfo(request, Lists.singleton(vo));
return vo;
}
@Override
public List<HostVO> getHostList(HostQueryRequest request) {
try {
// 条件
LambdaQueryWrapper<HostDO> wrapper = this.buildQueryWrapper(request);
if (wrapper == null) {
return Lists.empty();
}
// 查询
List<HostVO> hosts = hostDAO.of(wrapper).list(HostConvert.MAPPER::to);
// 查询拓展信息
this.setExtraInfo(request, hosts);
return hosts;
} finally {
FAVORITE_HOLDER.remove();
}
}
@Override
public DataGrid<HostVO> getHostPage(HostQueryRequest request) {
try {
// 条件
LambdaQueryWrapper<HostDO> wrapper = this.buildQueryWrapper(request);
if (wrapper == null) {
return DataGrid.of(Lists.empty());
}
// 查询
DataGrid<HostVO> hosts = hostDAO.of(wrapper)
.page(request)
.dataGrid(HostConvert.MAPPER::to);
// 查询拓展信息
this.setExtraInfo(request, hosts.getRows());
return hosts;
} finally {
FAVORITE_HOLDER.remove();
}
}
@Override
public Integer deleteHostById(Long id) {
log.info("HostService-deleteHostById id: {}", id);
int effect = hostDAO.deleteById(id);
log.info("HostService-deleteHostById effect: {}", effect);
// 删除配置
hostConfigDAO.deleteByHostId(id);
// 删除 tag 引用
tagRelApi.deleteRelId(TagTypeEnum.HOST, id);
// 删除收藏引用
favoriteApi.deleteByRelId(FavoriteTypeEnum.HOST, id);
return effect;
}
/**
* 检测 name 是否存在
*
* @param domain domain
*/
private void checkHostNamePresent(HostDO domain) {
// 构造条件
LambdaQueryWrapper<HostDO> wrapper = hostDAO.wrapper()
// 更新时忽略当前记录
.ne(HostDO::getId, domain.getId())
.eq(HostDO::getName, domain.getName());
// 检查是否存在
boolean present = hostDAO.of(wrapper).present();
Valid.isFalse(present, ErrorMessage.NAME_PRESENT);
}
/**
* 检测 code 是否存在
*
* @param domain domain
*/
private void checkHostCodePresent(HostDO domain) {
// 构造条件
LambdaQueryWrapper<HostDO> wrapper = hostDAO.wrapper()
// 更新时忽略当前记录
.ne(HostDO::getId, domain.getId())
.eq(HostDO::getCode, domain.getCode());
// 检查是否存在
boolean present = hostDAO.of(wrapper).present();
Valid.isFalse(present, ErrorMessage.CODE_PRESENT);
}
/**
* 构建查询 wrapper
*
* @param request request
* @return wrapper
*/
@SneakyThrows
private LambdaQueryWrapper<HostDO> buildQueryWrapper(HostQueryRequest request) {
boolean setIdList = Booleans.isTrue(request.getFavorite()) || Lists.isNotEmpty(request.getTags());
List<List<Long>> idListGrouping = new ArrayList<>();
// 查询收藏
Future<List<Long>> favoriteFuture = null;
if (Booleans.isTrue(request.getFavorite())) {
favoriteFuture = favoriteApi.getFavoriteRelIdList(FavoriteTypeEnum.HOST, SecurityUtils.getLoginUserId());
}
// tag 条件
if (Lists.isNotEmpty(request.getTags())) {
List<Long> tagRelIdList = tagRelApi.getRelIdByTagId(request.getTags());
if (tagRelIdList.isEmpty()) {
return null;
}
idListGrouping.add(tagRelIdList);
}
// 获取收藏结果
if (favoriteFuture != null) {
List<Long> favorites = favoriteFuture.get();
// 无收藏
if (Lists.isEmpty(favorites)) {
return null;
}
idListGrouping.add(favorites);
}
// flat
List<Long> idList = null;
if (setIdList && !idListGrouping.isEmpty()) {
idList = idListGrouping.get(0);
// 交集
for (int i = 1; i < idListGrouping.size(); i++) {
idList.retainAll(idListGrouping.get(i));
}
if (idList.isEmpty()) {
return null;
}
}
// 基础条件
LambdaQueryWrapper<HostDO> wrapper = hostDAO.wrapper()
.eq(HostDO::getId, request.getId())
.like(HostDO::getName, request.getName())
.like(HostDO::getCode, request.getCode())
.like(HostDO::getAddress, request.getAddress());
if (setIdList) {
wrapper.in(HostDO::getId, idList);
}
return wrapper;
}
/**
* 设置额外信息
*
* @param request request
* @param hosts hosts
*/
@SneakyThrows
private void setExtraInfo(HostQueryRequest request, List<HostVO> hosts) {
if (hosts.isEmpty()) {
return;
}
List<Long> idList = hosts.stream().map(HostVO::getId).collect(Collectors.toList());
// 查询额外信息
Future<List<List<TagDTO>>> tagsFuture = null;
Future<List<Long>> favoriteFuture = null;
if (Booleans.isTrue(request.getExtra())) {
// tag
tagsFuture = tagRelApi.getRelTags(TagTypeEnum.HOST, idList);
// 从缓存中读取 收藏
List<Long> favorites = FAVORITE_HOLDER.get();
if (favorites != null) {
favoriteFuture = CompletableFuture.completedFuture(favorites);
} else {
favoriteFuture = favoriteApi.getFavoriteRelIdList(FavoriteTypeEnum.HOST, SecurityUtils.getLoginUserId());
}
}
// 查询配置
if (Booleans.isTrue(request.getConfig())) {
// 配置分组
Map<Long, List<HostConfigDO>> hostConfigGrouping = hostConfigDAO.getHostConfigByHostIdList(idList)
.stream()
.collect(Collectors.groupingBy(HostConfigDO::getHostId));
// 设置配置
hosts.forEach(s -> {
List<HostConfigDO> configs = hostConfigGrouping.get(s.getId());
if (Lists.isEmpty(configs)) {
return;
}
Map<String, HostConfigVO> configMap = configs.stream()
.collect(Collectors.toMap(
HostConfigDO::getType,
HostConfigConvert.MAPPER::to,
(v1, v2) -> v2
));
s.setConfigs(configMap);
});
}
// 设置 tag 信息
List<List<TagDTO>> tagList = null;
if (tagsFuture != null && (tagList = tagsFuture.get()) != null) {
for (int i = 0; i < hosts.size(); i++) {
List<TagDTO> tags = tagList.get(i);
if (Lists.isEmpty(tags)) {
continue;
}
Map<Long, String> tagMap = tags.stream()
.collect(Collectors.toMap(
TagDTO::getId,
TagDTO::getName,
(v1, v2) -> v2,
LinkedHashMap::new
));
hosts.get(i).setTags(tagMap);
}
}
// 设置收藏信息
List<Long> favoriteIdList = null;
if (favoriteFuture != null && (favoriteIdList = favoriteFuture.get()) != null) {
for (HostVO host : hosts) {
host.setFavorite(favoriteIdList.contains(host.getId()));
}
}
}
}

View File

@@ -0,0 +1,24 @@
<?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.asset.dao.HostConfigDAO">
<!-- 通用查询映射结果 -->
<resultMap id="BaseResultMap" type="com.orion.ops.module.asset.entity.domain.HostConfigDO">
<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="host_id" property="hostId"/>
<result column="type" property="type"/>
<result column="config" property="config"/>
<result column="version" property="version"/>
</resultMap>
<!-- 通用查询结果列 -->
<sql id="Base_Column_List">
id, host_id, type, config, version, create_time, update_time, creator, updater, deleted
</sql>
</mapper>

View File

@@ -0,0 +1,23 @@
<?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.asset.dao.HostDAO">
<!-- 通用查询映射结果 -->
<resultMap id="BaseResultMap" type="com.orion.ops.module.asset.entity.domain.HostDO">
<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="code" property="code"/>
<result column="address" property="address"/>
</resultMap>
<!-- 通用查询结果列 -->
<sql id="Base_Column_List">
id, name, code, address, create_time, update_time, creator, updater, deleted
</sql>
</mapper>