文件配置的本地缓存,使用 Job 轮询,替代 MQ 广播
This commit is contained in:
parent
91e0af0944
commit
ca886c2791
@ -6,6 +6,9 @@ import cn.iocoder.yudao.framework.mybatis.core.query.LambdaQueryWrapperX;
|
|||||||
import cn.iocoder.yudao.module.infra.controller.admin.file.vo.config.FileConfigPageReqVO;
|
import cn.iocoder.yudao.module.infra.controller.admin.file.vo.config.FileConfigPageReqVO;
|
||||||
import cn.iocoder.yudao.module.infra.dal.dataobject.file.FileConfigDO;
|
import cn.iocoder.yudao.module.infra.dal.dataobject.file.FileConfigDO;
|
||||||
import org.apache.ibatis.annotations.Mapper;
|
import org.apache.ibatis.annotations.Mapper;
|
||||||
|
import org.apache.ibatis.annotations.Select;
|
||||||
|
|
||||||
|
import java.time.LocalDateTime;
|
||||||
|
|
||||||
@Mapper
|
@Mapper
|
||||||
public interface FileConfigMapper extends BaseMapperX<FileConfigDO> {
|
public interface FileConfigMapper extends BaseMapperX<FileConfigDO> {
|
||||||
@ -18,4 +21,7 @@ public interface FileConfigMapper extends BaseMapperX<FileConfigDO> {
|
|||||||
.orderByDesc(FileConfigDO::getId));
|
.orderByDesc(FileConfigDO::getId));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Select("SELECT COUNT(*) FROM infra_file_config WHERE update_time > #{maxUpdateTime}")
|
||||||
|
Long selectCountByUpdateTimeGt(LocalDateTime maxUpdateTime);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,29 +0,0 @@
|
|||||||
package cn.iocoder.yudao.module.infra.mq.consumer.file;
|
|
||||||
|
|
||||||
import cn.iocoder.yudao.module.infra.mq.message.file.FileConfigRefreshMessage;
|
|
||||||
import cn.iocoder.yudao.module.infra.service.file.FileConfigService;
|
|
||||||
import lombok.extern.slf4j.Slf4j;
|
|
||||||
import org.springframework.context.event.EventListener;
|
|
||||||
import org.springframework.stereotype.Component;
|
|
||||||
|
|
||||||
import javax.annotation.Resource;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 针对 {@link FileConfigRefreshMessage} 的消费者
|
|
||||||
*
|
|
||||||
* @author 芋道源码
|
|
||||||
*/
|
|
||||||
@Component
|
|
||||||
@Slf4j
|
|
||||||
public class FileConfigRefreshConsumer {
|
|
||||||
|
|
||||||
@Resource
|
|
||||||
private FileConfigService fileConfigService;
|
|
||||||
|
|
||||||
@EventListener
|
|
||||||
public void execute(FileConfigRefreshMessage message) {
|
|
||||||
log.info("[execute][收到 FileConfig 刷新消息]");
|
|
||||||
fileConfigService.initLocalCache();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -0,0 +1,4 @@
|
|||||||
|
/**
|
||||||
|
* 消息队列的消费者
|
||||||
|
*/
|
||||||
|
package cn.iocoder.yudao.module.infra.mq.consumer;
|
@ -1,19 +0,0 @@
|
|||||||
package cn.iocoder.yudao.module.infra.mq.message.file;
|
|
||||||
|
|
||||||
import lombok.Data;
|
|
||||||
import org.springframework.cloud.bus.event.RemoteApplicationEvent;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 文件配置数据刷新 Message
|
|
||||||
*/
|
|
||||||
@Data
|
|
||||||
public class FileConfigRefreshMessage extends RemoteApplicationEvent {
|
|
||||||
|
|
||||||
public FileConfigRefreshMessage() {
|
|
||||||
}
|
|
||||||
|
|
||||||
public FileConfigRefreshMessage(Object source, String originService, String destinationService) {
|
|
||||||
super(source, originService, DEFAULT_DESTINATION_FACTORY.getDestination(destinationService));
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -0,0 +1,4 @@
|
|||||||
|
/**
|
||||||
|
* 消息队列的消息
|
||||||
|
*/
|
||||||
|
package cn.iocoder.yudao.module.infra.mq.message;
|
@ -1,20 +0,0 @@
|
|||||||
package cn.iocoder.yudao.module.infra.mq.producer.file;
|
|
||||||
|
|
||||||
import cn.iocoder.yudao.framework.mq.core.bus.AbstractBusProducer;
|
|
||||||
import cn.iocoder.yudao.module.infra.mq.message.file.FileConfigRefreshMessage;
|
|
||||||
import org.springframework.stereotype.Component;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 文件配置相关消息的 Producer
|
|
||||||
*/
|
|
||||||
@Component
|
|
||||||
public class FileConfigProducer extends AbstractBusProducer {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 发送 {@link FileConfigRefreshMessage} 消息
|
|
||||||
*/
|
|
||||||
public void sendFileConfigRefreshMessage() {
|
|
||||||
publishEvent(new FileConfigRefreshMessage(this, selfDestinationService(), selfDestinationService()));
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -0,0 +1,4 @@
|
|||||||
|
/**
|
||||||
|
* 消息队列的生产者
|
||||||
|
*/
|
||||||
|
package cn.iocoder.yudao.module.infra.mq.producer;
|
@ -16,11 +16,6 @@ import javax.validation.Valid;
|
|||||||
*/
|
*/
|
||||||
public interface FileConfigService {
|
public interface FileConfigService {
|
||||||
|
|
||||||
/**
|
|
||||||
* 初始化文件客户端
|
|
||||||
*/
|
|
||||||
void initLocalCache();
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 创建文件配置
|
* 创建文件配置
|
||||||
*
|
*
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
package cn.iocoder.yudao.module.infra.service.file;
|
package cn.iocoder.yudao.module.infra.service.file;
|
||||||
|
|
||||||
|
import cn.hutool.core.collection.CollUtil;
|
||||||
import cn.hutool.core.io.resource.ResourceUtil;
|
import cn.hutool.core.io.resource.ResourceUtil;
|
||||||
import cn.hutool.core.util.IdUtil;
|
import cn.hutool.core.util.IdUtil;
|
||||||
import cn.iocoder.yudao.framework.common.pojo.PageResult;
|
import cn.iocoder.yudao.framework.common.pojo.PageResult;
|
||||||
|
import cn.iocoder.yudao.framework.common.util.collection.CollectionUtils;
|
||||||
import cn.iocoder.yudao.framework.common.util.json.JsonUtils;
|
import cn.iocoder.yudao.framework.common.util.json.JsonUtils;
|
||||||
import cn.iocoder.yudao.framework.common.util.validation.ValidationUtils;
|
import cn.iocoder.yudao.framework.common.util.validation.ValidationUtils;
|
||||||
import cn.iocoder.yudao.framework.file.core.client.FileClient;
|
import cn.iocoder.yudao.framework.file.core.client.FileClient;
|
||||||
@ -15,20 +17,20 @@ import cn.iocoder.yudao.module.infra.controller.admin.file.vo.config.FileConfigU
|
|||||||
import cn.iocoder.yudao.module.infra.convert.file.FileConfigConvert;
|
import cn.iocoder.yudao.module.infra.convert.file.FileConfigConvert;
|
||||||
import cn.iocoder.yudao.module.infra.dal.dataobject.file.FileConfigDO;
|
import cn.iocoder.yudao.module.infra.dal.dataobject.file.FileConfigDO;
|
||||||
import cn.iocoder.yudao.module.infra.dal.mysql.file.FileConfigMapper;
|
import cn.iocoder.yudao.module.infra.dal.mysql.file.FileConfigMapper;
|
||||||
import cn.iocoder.yudao.module.infra.mq.producer.file.FileConfigProducer;
|
|
||||||
import lombok.Getter;
|
import lombok.Getter;
|
||||||
import lombok.extern.slf4j.Slf4j;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
|
import org.springframework.scheduling.annotation.Scheduled;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
import org.springframework.transaction.annotation.Transactional;
|
import org.springframework.transaction.annotation.Transactional;
|
||||||
import org.springframework.transaction.support.TransactionSynchronization;
|
|
||||||
import org.springframework.transaction.support.TransactionSynchronizationManager;
|
|
||||||
import org.springframework.validation.annotation.Validated;
|
import org.springframework.validation.annotation.Validated;
|
||||||
|
|
||||||
import javax.annotation.PostConstruct;
|
import javax.annotation.PostConstruct;
|
||||||
import javax.annotation.Resource;
|
import javax.annotation.Resource;
|
||||||
import javax.validation.Validator;
|
import javax.validation.Validator;
|
||||||
|
import java.time.LocalDateTime;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception;
|
import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception;
|
||||||
import static cn.iocoder.yudao.module.infra.enums.ErrorCodeConstants.FILE_CONFIG_DELETE_FAIL_MASTER;
|
import static cn.iocoder.yudao.module.infra.enums.ErrorCodeConstants.FILE_CONFIG_DELETE_FAIL_MASTER;
|
||||||
@ -46,6 +48,12 @@ public class FileConfigServiceImpl implements FileConfigService {
|
|||||||
|
|
||||||
@Resource
|
@Resource
|
||||||
private FileClientFactory fileClientFactory;
|
private FileClientFactory fileClientFactory;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 文件配置的缓存
|
||||||
|
*/
|
||||||
|
@Getter
|
||||||
|
private List<FileConfigDO> fileConfigCache;
|
||||||
/**
|
/**
|
||||||
* Master FileClient 对象,有且仅有一个,即 {@link FileConfigDO#getMaster()} 对应的
|
* Master FileClient 对象,有且仅有一个,即 {@link FileConfigDO#getMaster()} 对应的
|
||||||
*/
|
*/
|
||||||
@ -55,13 +63,9 @@ public class FileConfigServiceImpl implements FileConfigService {
|
|||||||
@Resource
|
@Resource
|
||||||
private FileConfigMapper fileConfigMapper;
|
private FileConfigMapper fileConfigMapper;
|
||||||
|
|
||||||
@Resource
|
|
||||||
private FileConfigProducer fileConfigProducer;
|
|
||||||
|
|
||||||
@Resource
|
@Resource
|
||||||
private Validator validator;
|
private Validator validator;
|
||||||
|
|
||||||
@Override
|
|
||||||
@PostConstruct
|
@PostConstruct
|
||||||
public void initLocalCache() {
|
public void initLocalCache() {
|
||||||
// 第一步:查询数据
|
// 第一步:查询数据
|
||||||
@ -76,6 +80,27 @@ public class FileConfigServiceImpl implements FileConfigService {
|
|||||||
masterFileClient = fileClientFactory.getFileClient(config.getId());
|
masterFileClient = fileClientFactory.getFileClient(config.getId());
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
this.fileConfigCache = configs;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 通过定时任务轮询,刷新缓存
|
||||||
|
*
|
||||||
|
* 目的:多节点部署时,通过轮询”通知“所有节点,进行刷新
|
||||||
|
*/
|
||||||
|
@Scheduled(initialDelay = 60, fixedRate = 60, timeUnit = TimeUnit.SECONDS)
|
||||||
|
public void refreshLocalCache() {
|
||||||
|
// 情况一:如果缓存里没有数据,则直接刷新缓存
|
||||||
|
if (CollUtil.isEmpty(fileConfigCache)) {
|
||||||
|
initLocalCache();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 情况二,如果缓存里数据,则通过 updateTime 判断是否有数据变更,有变更则刷新缓存
|
||||||
|
LocalDateTime maxTime = CollectionUtils.getMaxValue(fileConfigCache, FileConfigDO::getUpdateTime);
|
||||||
|
if (fileConfigMapper.selectCountByUpdateTimeGt(maxTime) > 0) {
|
||||||
|
initLocalCache();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -85,9 +110,9 @@ public class FileConfigServiceImpl implements FileConfigService {
|
|||||||
.setConfig(parseClientConfig(createReqVO.getStorage(), createReqVO.getConfig()))
|
.setConfig(parseClientConfig(createReqVO.getStorage(), createReqVO.getConfig()))
|
||||||
.setMaster(false); // 默认非 master
|
.setMaster(false); // 默认非 master
|
||||||
fileConfigMapper.insert(fileConfig);
|
fileConfigMapper.insert(fileConfig);
|
||||||
// 发送刷新配置的消息
|
|
||||||
fileConfigProducer.sendFileConfigRefreshMessage();
|
// 刷新缓存
|
||||||
// 返回
|
initLocalCache();
|
||||||
return fileConfig.getId();
|
return fileConfig.getId();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -99,8 +124,9 @@ public class FileConfigServiceImpl implements FileConfigService {
|
|||||||
FileConfigDO updateObj = FileConfigConvert.INSTANCE.convert(updateReqVO)
|
FileConfigDO updateObj = FileConfigConvert.INSTANCE.convert(updateReqVO)
|
||||||
.setConfig(parseClientConfig(config.getStorage(), updateReqVO.getConfig()));
|
.setConfig(parseClientConfig(config.getStorage(), updateReqVO.getConfig()));
|
||||||
fileConfigMapper.updateById(updateObj);
|
fileConfigMapper.updateById(updateObj);
|
||||||
// 发送刷新配置的消息
|
|
||||||
fileConfigProducer.sendFileConfigRefreshMessage();
|
// 刷新缓存
|
||||||
|
initLocalCache();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -112,15 +138,9 @@ public class FileConfigServiceImpl implements FileConfigService {
|
|||||||
fileConfigMapper.updateBatch(new FileConfigDO().setMaster(false));
|
fileConfigMapper.updateBatch(new FileConfigDO().setMaster(false));
|
||||||
// 更新
|
// 更新
|
||||||
fileConfigMapper.updateById(new FileConfigDO().setId(id).setMaster(true));
|
fileConfigMapper.updateById(new FileConfigDO().setId(id).setMaster(true));
|
||||||
// 发送刷新配置的消息
|
|
||||||
TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronization() {
|
|
||||||
|
|
||||||
@Override
|
// 刷新缓存
|
||||||
public void afterCommit() {
|
initLocalCache();
|
||||||
fileConfigProducer.sendFileConfigRefreshMessage();
|
|
||||||
}
|
|
||||||
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private FileClientConfig parseClientConfig(Integer storage, Map<String, Object> config) {
|
private FileClientConfig parseClientConfig(Integer storage, Map<String, Object> config) {
|
||||||
@ -143,8 +163,9 @@ public class FileConfigServiceImpl implements FileConfigService {
|
|||||||
}
|
}
|
||||||
// 删除
|
// 删除
|
||||||
fileConfigMapper.deleteById(id);
|
fileConfigMapper.deleteById(id);
|
||||||
// 发送刷新配置的消息
|
|
||||||
fileConfigProducer.sendFileConfigRefreshMessage();
|
// 刷新缓存
|
||||||
|
initLocalCache();
|
||||||
}
|
}
|
||||||
|
|
||||||
private FileConfigDO validateFileConfigExists(Long id) {
|
private FileConfigDO validateFileConfigExists(Long id) {
|
||||||
|
@ -16,7 +16,6 @@ import cn.iocoder.yudao.module.infra.controller.admin.file.vo.config.FileConfigP
|
|||||||
import cn.iocoder.yudao.module.infra.controller.admin.file.vo.config.FileConfigUpdateReqVO;
|
import cn.iocoder.yudao.module.infra.controller.admin.file.vo.config.FileConfigUpdateReqVO;
|
||||||
import cn.iocoder.yudao.module.infra.dal.dataobject.file.FileConfigDO;
|
import cn.iocoder.yudao.module.infra.dal.dataobject.file.FileConfigDO;
|
||||||
import cn.iocoder.yudao.module.infra.dal.mysql.file.FileConfigMapper;
|
import cn.iocoder.yudao.module.infra.dal.mysql.file.FileConfigMapper;
|
||||||
import cn.iocoder.yudao.module.infra.mq.producer.file.FileConfigProducer;
|
|
||||||
import lombok.Data;
|
import lombok.Data;
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
import org.springframework.boot.test.mock.mockito.MockBean;
|
import org.springframework.boot.test.mock.mockito.MockBean;
|
||||||
@ -55,8 +54,6 @@ public class FileConfigServiceImplTest extends BaseDbUnitTest {
|
|||||||
@Resource
|
@Resource
|
||||||
private FileConfigMapper fileConfigMapper;
|
private FileConfigMapper fileConfigMapper;
|
||||||
|
|
||||||
@MockBean
|
|
||||||
private FileConfigProducer fileConfigProducer;
|
|
||||||
@MockBean
|
@MockBean
|
||||||
private Validator validator;
|
private Validator validator;
|
||||||
@MockBean
|
@MockBean
|
||||||
@ -81,6 +78,10 @@ public class FileConfigServiceImplTest extends BaseDbUnitTest {
|
|||||||
verify(fileClientFactory).createOrUpdateFileClient(eq(2L),
|
verify(fileClientFactory).createOrUpdateFileClient(eq(2L),
|
||||||
eq(configDO2.getStorage()), eq(configDO2.getConfig()));
|
eq(configDO2.getStorage()), eq(configDO2.getConfig()));
|
||||||
assertSame(masterFileClient, fileConfigService.getMasterFileClient());
|
assertSame(masterFileClient, fileConfigService.getMasterFileClient());
|
||||||
|
// 断言 fileConfigCache 缓存
|
||||||
|
assertEquals(2, fileConfigService.getFileConfigCache().size());
|
||||||
|
assertEquals(configDO1, fileConfigService.getFileConfigCache().get(0));
|
||||||
|
assertEquals(configDO2, fileConfigService.getFileConfigCache().get(1));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -101,8 +102,6 @@ public class FileConfigServiceImplTest extends BaseDbUnitTest {
|
|||||||
assertFalse(fileConfig.getMaster());
|
assertFalse(fileConfig.getMaster());
|
||||||
assertEquals("/yunai", ((LocalFileClientConfig) fileConfig.getConfig()).getBasePath());
|
assertEquals("/yunai", ((LocalFileClientConfig) fileConfig.getConfig()).getBasePath());
|
||||||
assertEquals("https://www.iocoder.cn", ((LocalFileClientConfig) fileConfig.getConfig()).getDomain());
|
assertEquals("https://www.iocoder.cn", ((LocalFileClientConfig) fileConfig.getConfig()).getDomain());
|
||||||
// verify 调用
|
|
||||||
verify(fileConfigProducer).sendFileConfigRefreshMessage();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -126,8 +125,6 @@ public class FileConfigServiceImplTest extends BaseDbUnitTest {
|
|||||||
assertPojoEquals(reqVO, fileConfig, "config");
|
assertPojoEquals(reqVO, fileConfig, "config");
|
||||||
assertEquals("/yunai2", ((LocalFileClientConfig) fileConfig.getConfig()).getBasePath());
|
assertEquals("/yunai2", ((LocalFileClientConfig) fileConfig.getConfig()).getBasePath());
|
||||||
assertEquals("https://doc.iocoder.cn", ((LocalFileClientConfig) fileConfig.getConfig()).getDomain());
|
assertEquals("https://doc.iocoder.cn", ((LocalFileClientConfig) fileConfig.getConfig()).getDomain());
|
||||||
// verify 调用
|
|
||||||
verify(fileConfigProducer).sendFileConfigRefreshMessage();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -152,8 +149,6 @@ public class FileConfigServiceImplTest extends BaseDbUnitTest {
|
|||||||
// 断言数据
|
// 断言数据
|
||||||
assertTrue(fileConfigMapper.selectById(dbFileConfig.getId()).getMaster());
|
assertTrue(fileConfigMapper.selectById(dbFileConfig.getId()).getMaster());
|
||||||
assertFalse(fileConfigMapper.selectById(masterFileConfig.getId()).getMaster());
|
assertFalse(fileConfigMapper.selectById(masterFileConfig.getId()).getMaster());
|
||||||
// verify 调用
|
|
||||||
verify(fileConfigProducer).sendFileConfigRefreshMessage();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -174,8 +169,6 @@ public class FileConfigServiceImplTest extends BaseDbUnitTest {
|
|||||||
fileConfigService.deleteFileConfig(id);
|
fileConfigService.deleteFileConfig(id);
|
||||||
// 校验数据不存在了
|
// 校验数据不存在了
|
||||||
assertNull(fileConfigMapper.selectById(id));
|
assertNull(fileConfigMapper.selectById(id));
|
||||||
// verify 调用
|
|
||||||
verify(fileConfigProducer).sendFileConfigRefreshMessage();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
Loading…
Reference in New Issue
Block a user