Nacos作为SpringBoot服务的注册中心和配置中心。
在NacosServer中修改配置文件,在SpringBoot不重启的情况下,获取到修改的内容。
本例将在配置文件中配置一个 cml.age=100 的配置项,程序中编写一个方法读取配置文件,并通过 Get—>/test/age 接口提供给浏览器访问。
pom中引入 nacos 相关配置:discovery,config,bootstrap
网上有人说,需要引入 actuator ,其实不用。
org.springframework.cloud
spring-cloud-dependencies
${spring.cloud.dependencies}
pom
import
com.alibaba.cloud
spring-cloud-alibaba-dependencies
${spring.cloud.alibaba.dependencies}
pom
import
com.alibaba.cloud
spring-cloud-starter-alibaba-nacos-discovery
com.alibaba.cloud
spring-cloud-starter-alibaba-nacos-config
org.springframework.cloud
spring-cloud-starter-bootstrap
server:
port: 9556
spring:
application:
name: app
profiles:
active: test
nacos:
discovery:
username: nacos
password: nacos
server-addr: 192.168.1.61:8848
config:
server-addr: 192.168.1.61:8848
file-extension: yaml
此配置指 NacosServer 中的配置文件 app-dev.yml ,仅截取 cml.age 部分
cml:
age: 100
问题:RefreshScope注解为什么一定要添加在 controller 上面?为什么在主启动类上面添加不生效
@RefreshScope
@Api(tags = "测试 - api")
@Slf4j
@RestController
@RequestMapping("/test")
public class TestController {
/**
* 获取配置文件中的 cml.age 内容,若未获取到,默认值为18
*/
@Value("${cml.age:18}")
private String age;
@ApiOperation(value = "获取年龄 - 测试配置自动刷新", notes = "获取年龄 - 测试配置自动刷新")
@GetMapping("/age")
public String getAge() {
return age;
}
}
开启 nacos-refresh 日志,打印配置内容更新情况
logging:
level:
com.alibaba.cloud.nacos.refresh: debug
打印的日志:
2022-01-28 13:43:30.574 [com.alibaba.nacos.client.Worker.longPolling.fixed-192.168.1.61_8848-zjrkm-admin] DEBUG com.alibaba.cloud.nacos.refresh.NacosContextRefresher.innerReceive:136 - Refresh Nacos config group=DEFAULT_GROUP,dataId=identityQrCodeAdmin-service-cml-test.yaml,configInfo=spring:
application:
name:
.
.
.
.
.
在不重启SpringBoot服务的情况,多次在 NacosServer 中修改 cml.age 配置项的值,然后通过浏览器访问 /test/age 接口,发现每次都可以获取到最新的 age 值。
当项目集成了 SpringSecurity 的时候,如果在 UserDetailsService 实现类上添加 RefreshScope 注解,并在类中直接使用 @Value 获取配置文件的内容,会导致用户在登录的时候找不到该类,从而无法调用loadUserByUsername 方法做用户名密码校验,进而登录功能无法使用。
解决方案是:
核心代码:
Account 类:
import lombok.Data;
import lombok.ToString;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.cloud.context.config.annotation.RefreshScope;
import org.springframework.stereotype.Component;
import java.io.Serializable;
@RefreshScope
@Component
@ConfigurationProperties(prefix = "cml.admin")
@Data
@ToString
public class Account implements Serializable {
private String defaultUserName;
private String defaultPassword;
private String defaultUserName1;
private String defaultPassword1;
}
UserDetailsService 实现类:
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.cloud.context.config.annotation.RefreshScope;
import org.springframework.security.core.authority.AuthorityUtils;
import org.springframework.security.core.userdetails.UserDetails;
import org.springframework.security.core.userdetails.UserDetailsService;
import org.springframework.security.core.userdetails.UsernameNotFoundException;
import org.springframework.security.crypto.password.PasswordEncoder;
import org.springframework.stereotype.Service;
@Service
public class UserService implements UserDetailsService {
@Autowired
private PasswordEncoder passwordEncoder;
@Autowired
private Account account;
@Override
public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
if (account.getDefaultUserName().equals(username)) {
String password = passwordEncoder.encode(account.getDefaultPassword());
return new User(username, password, AuthorityUtils.commaSeparatedStringToAuthorityList("admin"));
}
if (account.getDefaultUserName1().equals(username)) {
String password = passwordEncoder.encode(account.getDefaultPassword1());
return new User(username, password, AuthorityUtils.commaSeparatedStringToAuthorityList("admin1"));
}
return null;
}
}
邮箱:[email protected]
技术交流QQ群:1158377441