Skip to content

Commit effe54e

Browse files
adesjardinmanikmagarAnil Konakalla
authored
feat: capture message attributes and text for flow logs (#79)
* feat: capture message attributes and text for flow logs Closes #78 --------- Co-authored-by: Manik Magar <[email protected]> Co-authored-by: Anil Konakalla <[email protected]>
1 parent 43b96d8 commit effe54e

10 files changed

+216
-16
lines changed

.gitignore

+4-1
Original file line numberDiff line numberDiff line change
@@ -3,4 +3,7 @@
33
target/
44
*.iml
55
anypoint-pom.xml
6-
unit-test.log
6+
unit-test.log
7+
8+
docs/
9+
.flattened-pom.xml

pom.xml

+1-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@
1010
</parent>
1111

1212
<artifactId>mule-custom-logger</artifactId>
13-
<version>3.0.1-SNAPSHOT</version>
13+
<version>3.1.0-SNAPSHOT</version>
1414
<packaging>mule-extension</packaging>
1515
<name>Mule Custom Logger</name>
1616
<description>Mule Custom Logger module that provides standard structured logging</description>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
package com.avioconsulting.mule.logger.api.processor;
2+
3+
import org.mule.runtime.extension.api.annotation.Alias;
4+
import org.mule.runtime.extension.api.annotation.param.Optional;
5+
import org.mule.runtime.extension.api.annotation.param.Parameter;
6+
import org.mule.runtime.extension.api.annotation.param.ParameterGroup;
7+
import org.mule.runtime.extension.api.annotation.param.display.Summary;
8+
9+
@Alias("flow-log-config")
10+
public class FlowLogConfig {
11+
12+
@Parameter
13+
@Summary("Name of the flow to associate given expression as attributes")
14+
private String flowName;
15+
16+
@Parameter
17+
@Optional
18+
@Summary("A valid dataweave expression that resolves to a Map object with key-value pairs")
19+
private String attributesExpressionText;
20+
21+
@Parameter
22+
@Optional
23+
@Summary("A valid dataweave expression that results in a String to append to default flow start message")
24+
private String messageExpressionText;
25+
26+
public String getAttributesExpressionText() {
27+
return attributesExpressionText;
28+
}
29+
30+
public void setAttributesExpressionText(String attributesExpressionText) {
31+
this.attributesExpressionText = attributesExpressionText;
32+
}
33+
34+
public String getMessageExpressionText() {
35+
return messageExpressionText;
36+
}
37+
38+
public void setMessageExpressionText(String messageExpressionText) {
39+
this.messageExpressionText = messageExpressionText;
40+
}
41+
42+
public String getFlowName() {
43+
return flowName;
44+
}
45+
46+
public FlowLogConfig setFlowName(String flowName) {
47+
this.flowName = flowName;
48+
return this;
49+
}
50+
51+
}

src/main/java/com/avioconsulting/mule/logger/api/processor/MessageAttributes.java

+4
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,10 @@ public List<MessageAttribute> getAttributeList() {
4141
return this.messageAttributes;
4242
}
4343

44+
public void addAttributes(Map<String, String> attributes) {
45+
attributes.forEach((key, value) -> messageAttributes.add(new MessageAttribute(key, value)));
46+
}
47+
4448
public Map<String, String> getAttributes() {
4549
Map<String, String> attributes = new LinkedHashMap<>();
4650
if (messageAttributes != null) {

src/main/java/com/avioconsulting/mule/logger/internal/config/CustomLoggerConfiguration.java

+47
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44

55
import com.avioconsulting.mule.logger.api.processor.Compressor;
66
import com.avioconsulting.mule.logger.api.processor.EncryptionAlgorithm;
7+
import com.avioconsulting.mule.logger.api.processor.FlowLogConfig;
78
import com.avioconsulting.mule.logger.api.processor.LogProperties;
89
import com.avioconsulting.mule.logger.internal.CustomLogger;
910
import com.avioconsulting.mule.logger.internal.CustomLoggerOperation;
@@ -18,14 +19,21 @@
1819
import org.mule.runtime.api.lifecycle.Startable;
1920
import org.mule.runtime.api.meta.ExpressionSupport;
2021
import org.mule.runtime.api.notification.NotificationListenerRegistry;
22+
import org.mule.runtime.core.api.el.ExpressionManager;
2123
import org.mule.runtime.extension.api.annotation.Expression;
2224
import org.mule.runtime.extension.api.annotation.Operations;
25+
import org.mule.runtime.extension.api.annotation.param.NullSafe;
2326
import org.mule.runtime.extension.api.annotation.param.Optional;
2427
import org.mule.runtime.extension.api.annotation.param.Parameter;
2528
import org.mule.runtime.extension.api.annotation.param.display.*;
2629
import org.mule.runtime.extension.api.client.ExtensionsClient;
2730
import org.slf4j.LoggerFactory;
2831

32+
import java.util.List;
33+
import java.util.Map;
34+
import java.util.function.Function;
35+
import java.util.stream.Collectors;
36+
2937
/**
3038
* This class represents an extension configuration, values set in this class
3139
* are commonly used across multiple
@@ -83,6 +91,14 @@ public class CustomLoggerConfiguration implements Startable, Initialisable {
8391
@Expression(ExpressionSupport.NOT_SUPPORTED)
8492
private LogProperties.LogLevel flowLogLevel;
8593

94+
@Parameter
95+
@DisplayName("Flow Log Attributes")
96+
@Summary("The level flow logs will be logged at if enabled")
97+
@NullSafe
98+
@Optional
99+
@Expression(ExpressionSupport.NOT_SUPPORTED)
100+
private List<FlowLogConfig> flowLogConfigs;
101+
86102
@Parameter
87103
@DisplayName("Flow Log Category Suffix")
88104
@Summary("This category will be appended to the default logger category and used for all flow logs")
@@ -138,6 +154,10 @@ public class CustomLoggerConfiguration implements Startable, Initialisable {
138154
@Inject
139155
ExtensionsClient extensionsClient;
140156

157+
@Inject
158+
ExpressionManager expressionManager;
159+
private Map<String, FlowLogConfig> flowLogConfigMap;
160+
141161
/**
142162
* Default constructor for auto-initialization
143163
*/
@@ -175,6 +195,15 @@ public CustomLoggerConfiguration(CustomLoggerRegistrationService customLoggerReg
175195

176196
private static boolean isNotificationListenerRegistered = false;
177197

198+
public Map<String, FlowLogConfig> getFlowLogConfigMap() {
199+
return flowLogConfigMap;
200+
}
201+
202+
public CustomLoggerConfiguration setFlowLogConfigs(List<FlowLogConfig> flowLogConfigs) {
203+
this.flowLogConfigs = flowLogConfigs;
204+
return this;
205+
}
206+
178207
public String getApplicationName() {
179208
return applicationName;
180209
}
@@ -291,6 +320,10 @@ public ExtensionsClient getExtensionsClient() {
291320
return extensionsClient;
292321
}
293322

323+
public ExpressionManager getExpressionManager() {
324+
return expressionManager;
325+
}
326+
294327
/**
295328
* This method is invoked by the MuleSoft application when the AVIO Custom
296329
* Logger is invoked to create the connection.
@@ -312,6 +345,8 @@ public void start() throws MuleException {
312345
customLoggerRegistrationService.setConfig(this);
313346
if (isEnableFlowLogs()) {
314347
classLogger.info("Flow logs enabled");
348+
flowLogConfigMap = flowLogConfigs.stream().collect(
349+
Collectors.toMap(FlowLogConfig::getFlowName, Function.identity()));
315350
synchronized (CustomLoggerConfiguration.class) {
316351
if (!isNotificationListenerRegistered) {
317352
classLogger.info("Creating and registering notification listener");
@@ -345,5 +380,17 @@ public void initialise() throws InitialisationException {
345380
throw new InitialisationException(createStaticMessage(
346381
"Encryption Algorithm must be provided if encryption password is being supplied"), this);
347382
}
383+
flowLogConfigs.forEach(flowLogConfig -> {
384+
if (flowLogConfig.getMessageExpressionText() == null
385+
&& flowLogConfig.getAttributesExpressionText() == null) {
386+
try {
387+
throw new InitialisationException(createStaticMessage(
388+
"Both 'attributesExpressionText' and 'messageExpressionText' cannot be empty, at least one or both must be specified."),
389+
this);
390+
} catch (InitialisationException e) {
391+
throw new RuntimeException(e);
392+
}
393+
}
394+
});
348395
}
349396
}

src/main/java/com/avioconsulting/mule/logger/internal/listeners/CustomLoggerAbstractNotificationListener.java

+65-6
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,22 @@
11
package com.avioconsulting.mule.logger.internal.listeners;
22

3-
import com.avioconsulting.mule.logger.api.processor.AdditionalProperties;
4-
import com.avioconsulting.mule.logger.api.processor.ExceptionProperties;
5-
import com.avioconsulting.mule.logger.api.processor.LogProperties;
6-
import com.avioconsulting.mule.logger.api.processor.MessageAttributes;
3+
import com.avioconsulting.mule.logger.api.processor.*;
74
import com.avioconsulting.mule.logger.internal.CustomLogger;
85
import com.avioconsulting.mule.logger.internal.config.CustomLoggerConfiguration;
96
import org.mule.runtime.api.component.location.ComponentLocation;
107
import org.mule.runtime.api.event.Event;
8+
import org.mule.runtime.api.metadata.TypedValue;
9+
import org.mule.runtime.api.notification.EnrichedServerNotification;
10+
11+
import java.util.Collections;
12+
import java.util.List;
13+
import java.util.Map;
14+
import java.util.Optional;
15+
import java.util.stream.Collectors;
1116

1217
public abstract class CustomLoggerAbstractNotificationListener {
1318
protected final CustomLoggerConfiguration config;
19+
private Map<String, String> emptyAttributes = Collections.emptyMap();
1420

1521
public CustomLoggerAbstractNotificationListener(CustomLoggerConfiguration config) {
1622
this.config = config;
@@ -19,7 +25,7 @@ public CustomLoggerAbstractNotificationListener(CustomLoggerConfiguration config
1925
protected abstract org.slf4j.Logger getClassLogger();
2026

2127
protected void logMessage(ComponentLocation location, Event event, String logMessage, String categoryPrefix,
22-
LogProperties.LogLevel level) {
28+
LogProperties.LogLevel level, Map<String, String> additionalAttributes) {
2329
CustomLogger logger = config.getLogger();
2430
LogProperties logProperties = new LogProperties();
2531
MessageAttributes messageAttributes = new MessageAttributes();
@@ -28,6 +34,7 @@ protected void logMessage(ComponentLocation location, Event event, String logMes
2834
.getValue();
2935
messageAttributes.setOTelContextObject(oTelContextObject);
3036
}
37+
messageAttributes.addAttributes(additionalAttributes);
3138
ExceptionProperties exceptionProperties = new ExceptionProperties();
3239
AdditionalProperties additionalProperties = new AdditionalProperties();
3340
additionalProperties.setIncludeLocationInfo(true);
@@ -41,4 +48,56 @@ protected void logMessage(ComponentLocation location, Event event, String logMes
4148
logger.log(logProperties, messageAttributes, exceptionProperties, additionalProperties, config,
4249
location, correlationId);
4350
}
44-
}
51+
52+
protected Map<String, String> getFlowLogAttributes(EnrichedServerNotification notification) {
53+
Map<String, String> value = emptyAttributes;
54+
FlowLogConfig flowLogConfig;
55+
/**
56+
* Flow name can contain wildcard (*)
57+
* We only look for wildcard either starting of the string or ending of the
58+
* string
59+
* ex: mq-listener-* will look for all the flows that starts with mq-listener
60+
* ex: *-mq-flow will look for all the flows that ends with -mq-flow
61+
**/
62+
Optional<Map.Entry<String, FlowLogConfig>> matchedEntry = config.getFlowLogConfigMap().entrySet().stream()
63+
.filter(entry -> matchWildcard(entry.getKey(), notification.getResourceIdentifier()))
64+
.findFirst();
65+
if (matchedEntry.isPresent()) {
66+
flowLogConfig = matchedEntry.get().getValue();
67+
TypedValue<Map<String, String>> evaluate = (TypedValue<Map<String, String>>) config
68+
.getExpressionManager()
69+
.evaluate("#[" + flowLogConfig.getAttributesExpressionText() + "]",
70+
notification.getEvent().asBindingContext());
71+
value = evaluate.getValue();
72+
if (value == null)
73+
value = emptyAttributes;
74+
}
75+
return value;
76+
}
77+
78+
public boolean matchWildcard(String wildcardKey, String searchString) {
79+
// Trim the wildcard key
80+
String cleanWildcardKey = wildcardKey.trim();
81+
82+
// Exact match if no wildcards
83+
if (searchString.equalsIgnoreCase(wildcardKey)) {
84+
return true;
85+
}
86+
87+
// Handle start wildcard
88+
if (cleanWildcardKey.startsWith("*")) {
89+
String suffix = cleanWildcardKey.substring(1);
90+
return searchString.endsWith(suffix);
91+
}
92+
93+
// Handle end wildcard
94+
if (cleanWildcardKey.endsWith("*")) {
95+
String prefix = cleanWildcardKey.substring(0, cleanWildcardKey.length() - 1);
96+
return searchString.startsWith(prefix);
97+
}
98+
99+
// If wildcard key is just '*', match everything
100+
return cleanWildcardKey.equals("*");
101+
}
102+
103+
}

src/main/java/com/avioconsulting/mule/logger/internal/listeners/CustomLoggerFlowRefNotificationListener.java

+3-1
Original file line numberDiff line numberDiff line change
@@ -59,8 +59,9 @@ public void onNotification(MessageProcessorNotification notification) {
5959
return;
6060
}
6161
classLogger.info(message);
62+
Map<String, String> flowLogAttributes = getFlowLogAttributes(notification);
6263
logMessage(location, notification.getEvent(), message, FLOW_REF_CATEGORY_SUFFIX,
63-
config.getFlowLogLevel());
64+
config.getFlowLogLevel(), flowLogAttributes);
6465
} catch (Exception e) {
6566
classLogger.error("Error processing flow notification", e);
6667
}
@@ -69,4 +70,5 @@ public void onNotification(MessageProcessorNotification notification) {
6970
"Configuration hasn't been supplied to notification listener yet, flow logs won't be generated.");
7071
}
7172
}
73+
7274
}

src/main/java/com/avioconsulting/mule/logger/internal/listeners/CustomLoggerPipelineNotificationListener.java

+26-2
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,18 @@
11
package com.avioconsulting.mule.logger.internal.listeners;
22

3+
import com.avioconsulting.mule.logger.api.processor.FlowLogConfig;
34
import com.avioconsulting.mule.logger.internal.config.CustomLoggerConfiguration;
5+
import org.mule.runtime.api.metadata.TypedValue;
46
import org.mule.runtime.api.notification.PipelineMessageNotification;
57
import org.mule.runtime.api.notification.PipelineMessageNotificationListener;
68
import org.slf4j.Logger;
79
import org.slf4j.LoggerFactory;
810

11+
import java.util.List;
12+
import java.util.Map;
13+
import java.util.Optional;
14+
import java.util.stream.Collectors;
15+
916
/*
1017
* Listener for Mule notifications on flow start, end and completion.
1118
*/
@@ -35,10 +42,24 @@ public void onNotification(PipelineMessageNotification notification) {
3542
+ "]");
3643
if (config != null) {
3744
try {
45+
String msgToAppend = "";
46+
Optional<Map.Entry<String, FlowLogConfig>> matchedEntry = config.getFlowLogConfigMap().entrySet()
47+
.stream()
48+
.filter(entry -> matchWildcard(entry.getKey(), notification.getResourceIdentifier()))
49+
.findFirst();
50+
if (matchedEntry.isPresent()) {
51+
FlowLogConfig flowLogConfig = matchedEntry.get().getValue();
52+
TypedValue<String> evaluate = (TypedValue<String>) config
53+
.getExpressionManager()
54+
.evaluate("#[" + flowLogConfig.getMessageExpressionText() + "]",
55+
notification.getEvent().asBindingContext());
56+
msgToAppend = evaluate.getValue();
57+
}
3858
String message = "Event not processed yet, this should never be shown";
3959
switch (Integer.parseInt(notification.getAction().getIdentifier())) {
4060
case PipelineMessageNotification.PROCESS_START:
41-
message = "Flow [" + notification.getResourceIdentifier() + "]" + " start";
61+
message = "Flow [" + notification.getResourceIdentifier() + "]" + " start "
62+
+ (msgToAppend != null ? msgToAppend : "");
4263
break;
4364
case PipelineMessageNotification.PROCESS_COMPLETE:
4465
message = "Flow [" + notification.getResourceIdentifier() + "]" + " end";
@@ -49,9 +70,12 @@ public void onNotification(PipelineMessageNotification notification) {
4970
return;
5071
}
5172
classLogger.debug(message);
73+
Map<String, String> flowLogAttributes = getFlowLogAttributes(notification);
5274
logMessage(notification.getComponent().getLocation(), notification.getEvent(), message,
5375
config.getFlowCategorySuffix(),
54-
config.getFlowLogLevel());
76+
config.getFlowLogLevel(), flowLogAttributes);
77+
} catch (ClassCastException castException) {
78+
classLogger.error("Message expression text in flow-log-config needs to be a String", castException);
5579
} catch (Exception e) {
5680
classLogger.error("Error processing flow notification", e);
5781
}

src/test/java/com/avioconsulting/mule/logger/CustomLoggerArtifactTest.java

+3-2
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@
1313
import java.nio.file.Files;
1414
import java.nio.file.Path;
1515
import java.nio.file.Paths;
16-
import java.time.Duration;
16+
import java.util.Collections;
1717

1818
import static java.util.concurrent.TimeUnit.MILLISECONDS;
1919
import static java.util.concurrent.TimeUnit.SECONDS;
@@ -44,7 +44,8 @@ public void cleanup() throws Exception {
4444
@Test
4545
public void testLoggerConfigForCorrelationId() throws Exception {
4646
// TODO: Intercept logs and validate entries
47-
CoreEvent coreEvent = flowRunner("custom-logger-configFlow").run();
47+
CoreEvent coreEvent = flowRunner("custom-logger-configFlow")
48+
.withAttributes(Collections.singletonMap("some", "value")).run();
4849
Assert.assertNotNull(coreEvent);
4950
}
5051

0 commit comments

Comments
 (0)