Small question regarding MDC with Java please.
At first, I had very straightforward methods, some methods with many parameters (I shorten the list of parameters to keep things short just for this question, but please imagine a lot of parameters)
public String invokeMethodForPerson(int age, String name, boolean isCool, long distanceRun, double weight) {
LOGGER.info("begin to invoke method invokeMethodForPerson");
return methodForPerson(age, name, isCool, distanceRun, weight);
}
public String invokeMethodForCar(String model, boolean isElectric, long price, int numberOfDoors) {
LOGGER.info("begin to invoke method invokeMethodForCar");
return methodForCar(model, isElectric, price, numberOfDoors);
}
public String invokeMethodForFlower(String name, String color) {
LOGGER.info("begin to invoke method invokeMethodForFlower");
return methodForFlower(name, color);
}
(This question is not about how to refactor the long list of parameters)
Then, I wanted to leverage MDC. MDC is very helpful, it allow better search in log aggregator tools etc. Having them as first level, not inside the logger itself is very helpful.
Therefore, in order to leverage MDC, the code went from simple, to something now similar to this, here is what I tried.
public String invokeMethodForPerson(int age, String name, boolean isCool, long distanceRun, double weight) {
MDC.put("age", age);
MDC.put("name", name);
MDC.put("isCool", isCool);
MDC.put("distanceRun", distanceRun);
MDC.put("weight", weight);
LOGGER.info("begin to invoke method invokeMethodForPerson");
return methodForPerson(age, name, isCool, distanceRun, weight);
}
public String invokeMethodForCar(String model, boolean isElectric, long price, int numberOfDoors) {
MDC.put("model", model);
MDC.put("isElectric", isElectric);
MDC.put("price", price);
MDC.put("numberOfDoors", numberOfDoors);
LOGGER.info("begin to invoke method invokeMethodForCar");
return methodForCar(model, isElectric, price, numberOfDoors);
}
public String invokeMethodForFlower(String name, String color) {
MDC.put("name", name);
MDC.put("color", color);
LOGGER.info("begin to invoke method invokeMethodForFlower");
return methodForFlower(name, color);
}
Problem: now looking at the code, there is actually more lines of MDC.put()
than actual business logic code
Question: Is there a cleaner way to leveraging MDC, for many parameters, other than just adding tons of lines of MDC.put()
, if possible, using AOP/aspects/advice/pointcut please?
Thank you
You can create your own utils to work with MDC.
Example of usage:
More strict example:
Example using
Map.of
, but it does not support nullable values.UPDATE:
AOP solution via AspectJ
Add annotation
@LogMDC
for method and parameter target. The idea is to add the ability to put all method arguments or specific ones toMDC
Add aspect which will catch methods marked with
@LogMDC
annotation and perform storing arguments to theMDC
.Example of using. Put all parameters of method to MDC
Example of using. Put only specific parameters of the method to MDC, that marked with annotation.
Please note, AspecJ have a little performace impact
Performance penalty for using AspectJ
Performance impact of using aop
Performance: using AspectJ to log run time of all methods
Depends on your use cases you should decide what to use simple util method call or aop solution.