Introduction to the
This article starts with the third module, soul-client, which automatically registers routing-related data. This module explores the SoulSpringMvcClient annotation for HTTP service registration
An overview of
If you have used NGINX or run soul, you should know that routing and forwarding rules need to be configured. In soul gateway, there are two ways to configure the routing and forwarding rules. The second is to add annotations in the background service, so that it automatically register configuration
In the previous article, we also received a separate configuration for the Divide plug-in, which needs to be configured with selectors and rules. How are selectors and rules obtained and corresponding respectively? Let’s start a preliminary exploration next
Source code for the Debug
Looking for an entry point
This is the first analysis, there is no concept, we first check from the run log for clues. Running the soul-examples-http module, we find the following log printed
o.d.s.client.common.utils.RegisterUtils : http client register success: {"appName":"http","context":"/http","path":"/http/order/findById","pathDesc":"Find by Id ", "rpcType" : "HTTP", "the host" : "172.21.160.1", "port", 8188, "ruleName" : "HTTP / / order/findById", "enabled" : true, "registerMetaDat a":false} o.d.s.client.common.utils.RegisterUtils : http client register success: {"appName":"http","context":"/http","path":"/http/order/save","pathDesc":"Save Order ", "rpcType" : "HTTP", "the host" : "172.21.160.1", "port", 8188, "ruleName" : "HTTP / / order/save", "enabled" : true, "registerMetaData ":false}Copy the code
As you can see from the above log, this is a successful registration. It is done in RegisterUtils. Let’s search for this class.
public final class RegisterUtils {
/**
* call register api.
*
* @param json request body
* @param url url
* @param rpcTypeEnum rcp type
*/
public static void doRegister(final String json, final String url, final RpcTypeEnum rpcTypeEnum) {
try {
// Call a third-party HTTP client to send a POST request. The URL is soul-admin's address and interface
String result = OkHttpTools.getInstance().post(url, json);
if (AdminConstants.SUCCESS.equals(result)) {
log.info("{} client register success: {} ", rpcTypeEnum.getName(), json);
} else {
log.error("{} client register error: {} ", rpcTypeEnum.getName(), json); }}catch (IOException e) {
log.error("cannot register soul admin param, url: {}, request body: {}", url, json, e); }}}Copy the code
As you can see from the above function, this is the last stop of the registration configuration. We set breakpoints on the above function to trace the call stack
After a restart, the breakpoint is entered and the following class is called by the following function
public class SpringMvcClientBeanPostProcessor implements BeanPostProcessor {
@Override
public Object postProcessAfterInitialization(@NonNull final Object bean, @NonNull final String beanName) throws BeansException {
if(controller ! =null|| restController ! =null|| requestMapping ! =null) {
SoulSpringMvcClient clazzAnnotation = AnnotationUtils.findAnnotation(bean.getClass(), SoulSpringMvcClient.class);
String prePath = "";
if (Objects.nonNull(clazzAnnotation)) {
if (clazzAnnotation.path().indexOf("*") > 1) {
String finalPrePath = prePath;
executorService.execute(() -> RegisterUtils.doRegister(buildJsonParams(clazzAnnotation, finalPrePath), url,
RpcTypeEnum.HTTP));
returnbean; } prePath = clazzAnnotation.path(); }}returnbean; }}Copy the code
If the call stack is missing, we can set a breakpoint in the above function, restart the call stack, and find that it is spring-related. The BeanPostProcessor is the key, and will trigger the run after the Bean is initialized
The reference link is used by the Spring BeanPostProcessor interface
Register configuration logic
In SpringMvcClientBeanPostProcessor postProcessAfterInitialization breakpoint function, will find that each bean can trigger execution
I have guessed that the registration configuration information is executed using AOP, but only correctly, it should be used AOP and is configured at the time of initialization. I feel that THERE is a lot I don’t know about Spring, I will have to do some work later
In this class, we initialize the soul-admin background URL interface address
Let’s look at the registration logic:
- 1. If the class has one of the Controller, RestController, or RequestMapping annotations, the next Soul annotation will be registered for judgment
- 2. If there is a reference to Soul in the class
- If the annotation is a wildcard, register all interfaces
- If not, get the interface prefix, get all the methods, and register any methods that have Soul annotations on them
public class SpringMvcClientBeanPostProcessor implements BeanPostProcessor {
private final ThreadPoolExecutor executorService;
// http://localhost:9095/soul-client/springmvc-register
private final String url;
private final SoulSpringMvcConfig soulSpringMvcConfig;
/**
* Instantiates a new Soul client bean post processor.
*
* @param soulSpringMvcConfig the soul spring mvc config
*/
public SpringMvcClientBeanPostProcessor(final SoulSpringMvcConfig soulSpringMvcConfig) {
ValidateUtils.validate(soulSpringMvcConfig);
// amdinUrl:http://localhost:9095
// contextPath:/http
// appName:http
// full:false
// host:null
// port:8188
this.soulSpringMvcConfig = soulSpringMvcConfig;
url = soulSpringMvcConfig.getAdminUrl() + "/soul-client/springmvc-register";
executorService = new ThreadPoolExecutor(1.1.0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<>());
}
@Override
public Object postProcessAfterInitialization(@NonNull final Object bean, @NonNull final String beanName) throws BeansException {
if (soulSpringMvcConfig.isFull()) {
return bean;
}
// Determine if there is one of the following three HTTP annotations
Controller controller = AnnotationUtils.findAnnotation(bean.getClass(), Controller.class);
RestController restController = AnnotationUtils.findAnnotation(bean.getClass(), RestController.class);
RequestMapping requestMapping = AnnotationUtils.findAnnotation(bean.getClass(), RequestMapping.class);
if(controller ! =null|| restController ! =null|| requestMapping ! =null) {
SoulSpringMvcClient clazzAnnotation = AnnotationUtils.findAnnotation(bean.getClass(), SoulSpringMvcClient.class);
String prePath = "";
if (Objects.nonNull(clazzAnnotation)) {
// Class has annotations and is a wildcard, registered as a wildcard,RPCType is directly set to HTTP
if (clazzAnnotation.path().indexOf("*") > 1) {
String finalPrePath = prePath;
executorService.execute(() -> RegisterUtils.doRegister(buildJsonParams(clazzAnnotation, finalPrePath), url,
RpcTypeEnum.HTTP));
return bean;
}
prePath = clazzAnnotation.path();
}
// Go through all the methods and register the ones with Soul annotations
final Method[] methods = ReflectionUtils.getUniqueDeclaredMethods(bean.getClass());
for (Method method : methods) {
SoulSpringMvcClient soulSpringMvcClient = AnnotationUtils.findAnnotation(method, SoulSpringMvcClient.class);
if(Objects.nonNull(soulSpringMvcClient)) { String finalPrePath = prePath; executorService.execute(() -> RegisterUtils.doRegister(buildJsonParams(soulSpringMvcClient, finalPrePath), url, RpcTypeEnum.HTTP)); }}}return bean;
}
// Construct the request content
private String buildJsonParams(final SoulSpringMvcClient soulSpringMvcClient, final String prePath) {
String contextPath = soulSpringMvcConfig.getContextPath();
String appName = soulSpringMvcConfig.getAppName();
Integer port = soulSpringMvcConfig.getPort();
String path = contextPath + prePath + soulSpringMvcClient.path();
String desc = soulSpringMvcClient.desc();
String configHost = soulSpringMvcConfig.getHost();
String host = StringUtils.isBlank(configHost) ? IpUtils.getHost() : configHost;
String configRuleName = soulSpringMvcClient.ruleName();
String ruleName = StringUtils.isBlank(configRuleName) ? path : configRuleName;
SpringMvcRegisterDTO registerDTO = SpringMvcRegisterDTO.builder()
.context(contextPath)
.host(host)
.port(port)
.appName(appName)
.path(path)
.pathDesc(desc)
.rpcType(soulSpringMvcClient.rpcType())
.enabled(soulSpringMvcClient.enabled())
.ruleName(ruleName)
.registerMetaData(soulSpringMvcClient.registerMetaData())
.build();
returnOkHttpTools.getInstance().getGson().toJson(registerDTO); }}Copy the code
Register configuration information sources
Let’s take a quick look at the source of those registrations
Here is an interface that needs to be registered:
@RestController
@RequestMapping("/order")
@SoulSpringMvcClient(path = "/order")
public class OrderController {
@PostMapping("/save")
@SoulSpringMvcClient(path = "/save" , desc = "Save order")
public OrderDTO save(@RequestBody final OrderDTO orderDTO) {
orderDTO.setName("hello world save order");
returnorderDTO; }}Copy the code
This is the soul profile:
soul:
http:
adminUrl: http://localhost:9095
port: 8188
contextPath: /http
appName: http
full: false
Copy the code
AppName, Context, and port are all retrieved from the configuration
Path, pathDesc, RPCType, and so on are all obtained in the program annotations (SoulSpringMvcClient). Enabled defaults to true
{
"appName": "http"."context": "/http"."path": "/http/order/save"."pathDesc": "Save order"."rpcType": "http"."host": "172.21.160.1"."port": 8188."ruleName": "/http/order/save"."enabled": true."registerMetaData": false
}
Copy the code
Easy to have a look at the Soul – the Admin interface processing function, can be seen from the incoming data selector and the rules of processing, and we think about (it further code too much, don’t explanation, CRUD should all seem have no problem), it is important to note that in the process of handling will trigger event publishing, It is further verified that Controller is the trigger entry for data synchronization events
public class SoulClientRegisterServiceImpl implements SoulClientRegisterService {
@Override
@Transactional
public String registerSpringMvc(final SpringMvcRegisterDTO dto) {
if (dto.isRegisterMetaData()) {
MetaDataDO exist = metaDataMapper.findByPath(dto.getPath());
if(Objects.isNull(exist)) { saveSpringMvcMetaData(dto); }}// Process the selector
String selectorId = handlerSpringMvcSelector(dto);
// Handle rules
handlerSpringMvcRule(selectorId, dto);
returnSoulResultMessage.SUCCESS; }}Copy the code
conclusion
This article takes a look at soul-Client’s HTTP registration annotations and Outlines the process:
- 1. If the class has one of the Controller, RestController, or RequestMapping annotations, the next Soul annotation will be registered for judgment
- 2. If there is a reference to Soul in the class
- If the annotation is a wildcard, register all interfaces
- If not, get the interface prefix, get all the methods, and register any methods that have Soul annotations on them
Registration information is basically obtained from configuration files and annotations
The registered content is the corresponding selector and rule
Soul Gateway source code parsing article list
The Denver nuggets
Understanding and preliminary operation
- Soul Gateway source code parsing (a) overview
- Soul gateway source code analysis (two) the initial operation of the code
Request processing flow parsing
- Soul Gateway source code parsing (iii) request processing overview
- Soul Gateway source code parsing (4) Dubbo request overview
- Soul gateway source code analysis (five) request type exploration
- Soul Gateway source code analysis (six) Sofa request processing overview
- Soul gateway source code analysis (seven) limit the flow plug-in exploration
- Soul gateway source code analysis (eight) route matching exploration
- Soul gateway source code analysis (nine) plug-in configuration loading preliminary
- Soul gateway source code analysis (ten) custom simple plug-in preparation
- Soul gateway source code parsing (11) request processing summary
Data Synchronization parsing
- Soul gateway source code analysis (12) data synchronization
- Soul gateway source code parsing (thirteen) Websocket synchronization data -Bootstrap end
- Soul Gateway source code parsing (fourteen) HTTP data synchronization -Bootstrap end
- Zookeeper data synchronization -Bootstrap end
- Soul gateway source code analysis (sixteen) Nacos data synchronization example run
- Soul gateway source code parsing (seventeen) Nacos data synchronization parsing -Bootstrap end
- Soul gateway source code parsing (18) Zookeeper data synchronization preliminary -Admin end
- Soul Gateway source code parsing (19) Nacos data synchronization initialization fix -Admin end
- Soul gateway source code parsing (20) Websocket data synchronization -Admin end
- Soul Gateway source code parsing (21) HTTP long polling data synchronization -Admin end
- Soul gateway source code analysis (22) data synchronization summary
Soul – the Client module
- Soul gateway source code analysis (23) SoulSpringMvcClient annotations
One day
- HTTP parameter request error