GH-1174 Add support for configurable Message conversion helper

This commit includes only the interfaces and interaction callbacks. The rest of the implementation is provided by binders in s-c-stream
This commit is contained in:
Oleg Zhurakousky
2024-09-16 16:05:57 +02:00
parent 85a4fffcd4
commit 571783ffae
5 changed files with 87 additions and 5 deletions

View File

@@ -1439,7 +1439,7 @@ public class SimpleFunctionRegistry implements FunctionRegistry {
@SuppressWarnings("unchecked")
private Object convertOutputMessageIfNecessary(Object output, String expectedOutputContetntType) {
String contentType;
if (((Message) output).getHeaders().containsKey(MessageHeaders.CONTENT_TYPE)) {
if (this.isOutputTypeMessage() && ((Message) output).getHeaders().containsKey(MessageHeaders.CONTENT_TYPE)) {
contentType = ((Message) output).getHeaders().get(MessageHeaders.CONTENT_TYPE).toString();
}
else {

View File

@@ -129,7 +129,9 @@ public class ContextFunctionCatalogAutoConfiguration {
mcList.add(new StringMessageConverter());
mcList.add(new PrimitiveTypesFromStringMessageConverter(conversionService));
messageConverter = new SmartCompositeMessageConverter(mcList);
messageConverter = new SmartCompositeMessageConverter(mcList, () -> {
return context.getBeansOfType(MessageConverterHelper.class).values();
});
if (functionInvocationHelper instanceof CloudEventsFunctionInvocationHelper) {
((CloudEventsFunctionInvocationHelper) functionInvocationHelper).setMessageConverter(messageConverter);
}

View File

@@ -0,0 +1,46 @@
/*
* Copyright 2015-2021 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.springframework.cloud.function.context.config;
import org.springframework.messaging.Message;
/**
* @author Oleg Zhurakousky
*/
public interface MessageConverterHelper {
/**
* This method will be called by the framework in cases when a message failed to convert.
* It allows you to signal to the framework if such failure should be considered fatal or not.
*
* @param message failed message
* @return true if conversion failure must be considered fatal.
*/
default boolean shouldFailIfCantConvert(Message<?> message) {
return false;
}
/**
* This method will be called by the framework in cases when a single message within batch of messages failed to convert.
* It provides a place for providing post-processing logic before message converter returns.
*
* @param message failed message.
* @param index index of failed message within the batch
*/
default void postProcessBatchMessageOnFailure(Message<?> message, int index) {
}
}

View File

@@ -19,8 +19,10 @@ package org.springframework.cloud.function.context.config;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.function.Supplier;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
@@ -31,6 +33,7 @@ import org.springframework.messaging.Message;
import org.springframework.messaging.MessageHeaders;
import org.springframework.messaging.converter.AbstractMessageConverter;
import org.springframework.messaging.converter.CompositeMessageConverter;
import org.springframework.messaging.converter.MessageConversionException;
import org.springframework.messaging.converter.MessageConverter;
import org.springframework.messaging.converter.SmartMessageConverter;
import org.springframework.messaging.support.MessageBuilder;
@@ -48,13 +51,22 @@ public class SmartCompositeMessageConverter extends CompositeMessageConverter {
private Log logger = LogFactory.getLog(this.getClass());
private final Supplier<Collection<MessageConverterHelper>> messageConverterHelpersSupplier;
public SmartCompositeMessageConverter(Collection<MessageConverter> converters) {
this(converters, null);
}
public SmartCompositeMessageConverter(Collection<MessageConverter> converters, Supplier<Collection<MessageConverterHelper>> messageConverterHelpersSupplier) {
super(converters);
this.messageConverterHelpersSupplier = messageConverterHelpersSupplier;
}
@Override
@Nullable
public Object fromMessage(Message<?> message, Class<?> targetClass) {
Collection<MessageConverterHelper> messageConverterHelpers = this.messageConverterHelpersSupplier != null
? this.messageConverterHelpersSupplier.get() : Collections.emptyList();
for (MessageConverter converter : getConverters()) {
if (!(message.getPayload() instanceof byte[]) && targetClass.isInstance(message.getPayload()) && !(message.getPayload() instanceof Collection<?>)) {
return message.getPayload();
@@ -71,12 +83,15 @@ public class SmartCompositeMessageConverter extends CompositeMessageConverter {
}
}
}
this.failConversionIfNecessary(message, messageConverterHelpers);
return null;
}
@SuppressWarnings("unchecked")
@Override
public Object fromMessage(Message<?> message, Class<?> targetClass, @Nullable Object conversionHint) {
Collection<MessageConverterHelper> messageConverterHelpers = this.messageConverterHelpersSupplier != null
? this.messageConverterHelpersSupplier.get() : Collections.emptyList();
if (!(message.getPayload() instanceof byte[]) && targetClass.isInstance(message.getPayload()) && !(message.getPayload() instanceof Collection<?>)) {
return message.getPayload();
}
@@ -105,8 +120,12 @@ public class SmartCompositeMessageConverter extends CompositeMessageConverter {
}
}
}
if (!isConverted) {
this.postProcessBatchMessage(message, messageConverterHelpers, resultList.size());
this.failConversionIfNecessary(message, messageConverterHelpers);
}
}
result = resultList;
return resultList;
}
else {
for (MessageConverter converter : getConverters()) {
@@ -120,10 +139,25 @@ public class SmartCompositeMessageConverter extends CompositeMessageConverter {
}
}
}
this.failConversionIfNecessary(message, messageConverterHelpers);
return result;
}
private void failConversionIfNecessary(Message<?> message, Collection<MessageConverterHelper> messageConverterHelpers) {
for (MessageConverterHelper messageConverterHelper : messageConverterHelpers) {
if (messageConverterHelper.shouldFailIfCantConvert(message)) {
throw new MessageConversionException("Failed to convert Message: " + message
+ ". None of the available Message converters were able to convert this Message");
}
}
}
private void postProcessBatchMessage(Message<?> message, Collection<MessageConverterHelper> messageConverterHelpers, int index) {
for (MessageConverterHelper messageConverterHelper : messageConverterHelpers) {
messageConverterHelper.postProcessBatchMessageOnFailure(message, index);
}
}
@Override
@Nullable
public Message<?> toMessage(Object payload, @Nullable MessageHeaders headers) {