Event Driven lightweight message format
eventeria | spring-boot | spring-cloud-stream |
---|---|---|
1.3.0 | 3.4.x | 4.2.x |
1.2.2 | 3.2.x ~ 3.3.x | 4.1.x |
1.1.1 | 3.0.x ~ 3.1.x | 4.0.x |
1.0.1 | 2.7.x | 3.2.x |
dependencies {
implementation("com.navercorp.eventeria:spring-boot-eventeria:${version}")
}
@Configuration
public class MessageConfig {
@Bean
ChannelBindable channelBindable() {
return new ChannelBindable();
}
@Bean
@ConditionalOnMissingBean
ChannelBinder channelBinder(
SubscribableChannelBindingTargetFactory bindingTargetFactory,
ChannelBindable channelBindable
) {
return new DefaultChannelBinder(bindingTargetFactory, channelBindable);
}
}
- Register your spring-cloud-stream produce channel in application.yml or application.properties
spring: cloud: stream: bindings: outbound-kafka-channel-command: destination: command_topic producer: partition-count: 2
- Register beans of outbound
MessageChannel
,SpringMessagePublisher
,IntegrationFlow
. - Inject
SpringMessagePublisher
bean and publish messages.
@Service
public class ExampleService {
ExampleService(
@Qualified("springMessagePublisherBean") SpringMessagePublisher springMessagePublisher
) {
this.springMessagePublisher = springMessagePublisher;
}
public void publish(Message message) {
springMessagePublisher.publish(message);
}
}
NOTE: Does not support Functional publishing currently.
- First, register your spring-cloud-stream consume channel in application.yml or application.properties
spring: cloud: stream: bindings: inbound-kafka-channel-command: group: example_group destination: command_topic consumer: concurrency: 1
- Choose one of the Programmatic or Functional way to consume
Message
s.
- Register beans of inbound
SubscribableChannel
,SpringMessagePublisher
,IntegrationFlow
.MessageRouter
- example code
MessageRouter
: route messages to your application.
- Register additional function mappings in application.yml or application.properties.
spring: cloud: function: definition: transformCloudEventToMessage|consumeImplementsEventeriaMessage; transformCloudEventToMessage|consumeXXX;
- Register
Function
bean which converts spring-messagingMessage
to eventeriaMessage
.@Bean Function<org.springframework.messaging.Message<byte[]>, Message> transformCloudEventToMessage( CloudEventMessageReaderWriter cloudEventMessageReaderWriter ) { return FunctionalBindingSupports.convertToMessage(cloudEventMessageReaderWriter); }
- Register your
Consumer
bean which receives data type of eventeriaMessage
.@Bean Consumer<ImplementsEventeriaMessage> consumeImplementsEventeriaMessage() { return command -> log.info("[CONSUME][consumeImplementsEventeriaMessage] {}", command); }
Eventeria provides the typealias
extension to support following features.
- multiple data formats for single topic.
- publish the string of message type in fixed string, even if rename of class or relocate package.
- When publishing
Message
, the message will containtypealias
extension that you registered byaddSerializeTypeAlias
oraddCompatibleTypeAlias
. If not, full class name will be used.
- When consuming
Message
, eventeria try to deserialize the message withtypealias
extension. You should register typealias byaddDeserializeTypeAlias
or `addCompatibleTypeAlias - If eventeria failed to find deserialize target type, it try to load class using ClassLoader with provided
typealias
string. If there is no matching class, consuming will be failed.
@Bean
CloudEventMessageTypeAliasMapper cloudEventMessageTypeAliasMapper() {
CloudEventMessageTypeAliasMapper typeAliasMapper = new CloudEventMessageTypeAliasMapper();
// support typealias at both serialization / deserialization times
typeAliasMapper.addCompatibleTypeAlias(
PostCreatedEvent.class,
"com.navercorp.eventeria.guide.boot.domain.PostCreatedEvent"
);
// support typealias at serialization time only
typeAliasMapper.addSerializeTypeAlias(
SerializeOnlyTypeAliasPostCreatedEvent.class,
"com.navercorp.eventeria.guide.boot.domain.PostCreatedEventSerializeOnlyVersion"
);
return typeAliasMapper;
}
If typealias is configured correctly, you can also configure 'route functions' to handle message by class type.
This feature is provided for programmatic binding only.
// register router
@Bean("router")
MessagePayloadTypeRouter eventRouter(
YourEventHandler handler
) {
return MessagePayloadTypeRouter.register("beanname")
.route(CreatedEvent.class, handler::onCreatedEvent)
.route(RemovedEvent.class, handler::onRemovedEvent)
.done();
}
// map inbound handler and router
@Bean
IntegrationFlow eventRouterIntegrationFlow(
@Qualifier("inboundHandlerName") SpringMessageHandler springMessageHandler,
@Qualifier("router") MessagePayloadTypeRouter router
) {
return IntegrationFlow.from(springMessageHandler)
.route(router)
.get();
}
You can schedule the time when a message is published.
To use this feature, you should configure a persistence store to save/retrieve TimerMessage
.
- Publish a message that implements
TimerMessage
, with the time when should be published. - Published message is converted to
TimerMessageStoreValue
which is saved inTimerMessageStore
. - A retriever finds messages to publish from
TimerMessageStore
periodically.- If exists, it will publish them to actual output channel.
dependencies {
implementation("com.navercorp.eventeria:eventeria-timer-spring-integration:${version}")
// provides `TimerMessageStoreValue` implementations using redis
implementation("com.navercorp.eventeria:eventeria-timer-spring-redis:${version}")
// If you use in-memory or other persistence store, import this and implement `TimerMessageStoreValue` for your store.
implementation("com.navercorp.eventeria:eventeria-timer:${version}")
}
You must configure beans following, SpringTimerMessageHandler
, Scheduler
, IntegrationFlowAdapter
// bean example using in-memory store
@Bean
SpringTimerMessageHandler springTimerMessageHandler() {
TimerMessageStore timerMessageStore = new InMemoryTimerMessageStore();
TimerMessageHandler timerMessageHandler = new SimpleTimerMessageHandler(timerMessageStore, 1000);
return new SpringTimerMessageHandler(timerMessageHandler);
}
// bean example using redis store with distributed partitions supported.
@Bean
SpringTimerMessageHandler springTimerMessageHandler(
CloudEventMessageReaderWriter cloudEventMessageReaderWriter,
RedisOperations<String, String> redisIndexOperations,
RedisOperations<String, RedisCloudEventTimerMessageStoreValue> redisCloudEventTimerMessageStoreValueRedisOperations
) {
TimerMessageStore timerMessageStore = new RedisCloudEventTimerMessageStore(
"timer-message-", // redis key prefix
redisIndexOperations,
redisCloudEventTimerMessageStoreValueRedisOperations,
cloudEventMessageReaderWriter
);
TimerMessageHandler timerMessageHandler = new DistributedTimerMessageHandler(
timerMessageStore,
1000, // count per each release
DefaultPartitionGenerator(),
10, // registered partition count
10, // seek partition count
null // Executor
);
return new SpringTimerMessageHandler(timerMessageHandler);
}
@EnableScheduling
@Configuration
@RequiredArgsConstructor
public class SchedulingConfig {
private final SpringTimerMessageHandler springTimerMessageHandler;
@Scheduled(fixedDelay = 1000L)
void triggerReschedule() {
springTimerMessageHandler.reschedulePersistedMessages();
}
}
@Bean
IntegrationFlowAdapter outboundIntegrationFlowAdapter(
MessageChannel outboundChannel,
MessageToCloudEventConverter messageToCloudEventConverter,
CloudEventHeaderMapper cloudEventHeaderMapper,
SpringMessagePublisher springMessagePublisher,
SpringTimerMessageHandler springTimerMessageHandler
) {
return new TimerMessagePublisherIntegrationAdapter(
springMessagePublisher,
messageToCloudEventConverter,
cloudEventHeaderMapper,
outboundChannel,
springTimerMessageHandler
);
}
Now, you can use message scheduling like below.
public class ScheduledMessage implements TimerMessage {
}
springMessagePublisher.publish(new ScheduledMessage());
Copyright (c) 2022-present NAVER Corp.
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
http://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.