SlideShare a Scribd company logo
1 of 50
Download to read offline
A Deep Dive into

Spring Application Events
Oliver Drotbohm ƀ odrotbohm@vmware.com/ odrotbohm
fundamentalsProgramming model
transactionsConsistency & error scenarios
architectureComponents & relationships
2
fundamentalsProgramming model
3
ApplicationEventPublisher
ApplicationListener<T>
@EventListener
@TransactionalEventListener
Programming model
4
Application events fundamentals
DEMO
5
Application Events

with Spring Data
6
Application events with Spring (Data)
• Powerful mechanism to publish events in Spring applications
• Application event – either a general object or extending ApplicationEvent
• ApplicationEventPublisher – injectable to manually invoke event publication
• Spring Data event support
• Spring Data’s focus: aggregates and repositories
• Domain-Driven Design aggregates produce application events
• AbstractAggregateRoot<T> – base class to easily capture events and get them published on
CrudRepository.save(…) invocations.
• No dependency to infrastructure APIs
7
8
class Order {
private List<Object> events;
@DomainEvents
Collection<Object> events() { … }
@AfterDomainEventPublication
void reset() { … }
}
Ag#$eg&'( a+c-./0at23

ev256 7n36an92:
Met<=> t? e@A=s( 'Cem
Wip2 (HeIt3 &J($

puL0McN'i=I
Spring Data provided base class
9
class Order extends AbstractAggregateRoot<Order> {
Order complete() {
registerEvent(OrderCompleted.of(this));
return this;
}
}
Met<=> t? OePMs'($ ev256.

Sup2O +lN3: taT2: cNOe =U (xWo:-r(.
10
@Component
class OrderManagement {
private final OrderRepository orders;
@Transactional
void completeOrder(Order order) {

orders.save(order.complete());
}
}
No r2[($en92: t? SW$in#

in[$&s'$/c'u$2 AP]s
Intermediate summary
Events for Bounded Context interaction
Spring’s application events are a very light-weight way to implement those domain events. Spring Data
helps to easily expose them from aggregate roots. The overall pattern allows loosely coupled
interaction between Bounded Contexts so that the system can be extended and evolved easily.
Externalize events if needed
Depending on the integration mechanism that’s been selected we can now write separate components
to translate those JVM internal events into the technology of choice (JMS, AMQP, Kafka) to notify third-
party systems.
11
transactionsConsistency & error scenarios
14
15
@EventListener
@EventListener
…
@TransactionalEventListener
@TransactionalEventListener
…
@Transactional
ƻ

Event
ƻ

Event
Commit
1
2
3
4
5
6
7
8
Consistency boundary
Spring bean
Legend
Application events in a Spring application
1. We enter a transactional method
Business code is executed and might trigger state changes on aggregates.
2. That transactional method produces application events
In case the business code produces application events, standard events are published directly. For each
transactional event listener registered a transaction synchronization is registered, so that the event will
eventually be published on transaction completion (by default on transaction commit).
3. Event listeners are triggered
By default, event listeners are synchronously invoked, which means they participate in the currently
running transactions. This allows listeners to abort the overall transaction and ensure strong
consistency. Alternatively, listeners can be executed asynchronously using @Async. They then have to
take care of their transactional semantics themselves and errors will not break the original transaction.
16
Application events in a Spring application
4. Service execution proceeds once event delivery is completed
Once all standard event listeners have been invoked, the business logic is executed further. More events
can be published, further state changes can be created.
5. The transaction finishes
Once the transactional method is done, the transaction is completed. Usually all pending changes
(created by the main business code or the synchronous event listeners) are written to the database. In
case inconsistencies or connection problems, the transaction rolls back.
6. Transactional event listeners are triggered
Listeners annotated with @TransactionalEventListener are triggered when the transaction
commits, which means they can rely on the business operation the event has been triggered from
having succeeded. This allows the listeners to read committed data. Listeners can be invoked
asynchronously using @Async in case the functionality to be invoked might be long-running (e.g.
sending an email).
17
Error scenarios
18
What if the

service fails?
19
!
20
@EventListener
@EventListener
…
@TransactionalEventListener
@TransactionalEventListener
…
@Transactional
ƻ

Event
ƻ

Event
Commit
Consistency boundary
Spring bean
Legend
"#
21
@EventListener
@EventListener
…
@TransactionalEventListener
@TransactionalEventListener
…
@Transactional
ƻ

Event
ƻ

Event
Commit
Consistency boundary
Spring bean
Legend
Rollback!
$
"#
What if an event

listener fails?
22
!
23
@EventListener
@EventListener
…
@TransactionalEventListener
@TransactionalEventListener
…
@Transactional
ƻ

Event
ƻ

Event
Commit
Consistency boundary
Spring bean
Legend
"#
24
@EventListener
@EventListener
…
@TransactionalEventListener
@TransactionalEventListener
…
@Transactional
ƻ

Event
ƻ

Event
Commit
Consistency boundary
Spring bean
Legend
"#
Rollback!
$
What if a transactional

event listener fails?
25
!
26
@EventListener
@EventListener
…
@TransactionalEventListener
@TransactionalEventListener
…
@Transactional
ƻ

Event
ƻ

Event
Commit
Consistency boundary
Spring bean
Legend
"#
26
@EventListener
@EventListener
…
@TransactionalEventListener
@TransactionalEventListener
…
@Transactional
ƻ

Event
ƻ

Event
Commit
Consistency boundary
Spring bean
Legend


Publication
lost!
%"#
Transactional application events
DEMO
27
Application events – Error Scenarios
The service fails
Only standard event listeners up until the failure will have been executed. Assuming the already
triggered event listeners also execute transactional logic, the local transaction is rolled back and the
system is still in a strongly consistent state. Transactional event listeners are not invoked in the first
place.
A synchronous event listener fails
In case a normal event listener fails the entire transaction will roll back. This enables strong consistency
between the event producer and the listeners registered but also bears the risk of supporting
functionality interfering with the primary one, causing the latter to fail for less important reasons. The
tradeoff here could be to move to a transactional event listener and embrace eventual consistency.
28
Application events – Error Scenarios
A transactional event listener fails
In case a transactional event lister fails or the application crashes while transactional event listeners are
executed, the event is lost and functionality might not have been invoked. Other transactional event
listeners will still be triggered.
BONUS: An asynchronous event listener fails
The event is lost but the primary functionality can still succeed as the event is handled in a separate
thread. Retry mechanisms can (should?) be deployed in case some form of recovery is needed.
29
Event Publication Registry
30
31
@TransactionalEventListener
@TransactionalEventListener
…
@TransactionalEventListener
…
@TransactionalEventListener
…
31
@TransactionalEventListener
@TransactionalEventListener
…
@TransactionalEventListener
…
@TransactionalEventListener
…
31
@TransactionalEventListener
@TransactionalEventListener
…
ƻ

Event
@TransactionalEventListener
…
@TransactionalEventListener
…
Transaction Commit
31
@TransactionalEventListener
@TransactionalEventListener
…
ƻ

Event
@TransactionalEventListener
…
@TransactionalEventListener
…
Transaction Commit
31
@TransactionalEventListener
@TransactionalEventListener
…
ƻ

Event
@TransactionalEventListener
…
@TransactionalEventListener
…
Transaction Commit
31
@TransactionalEventListener
@TransactionalEventListener
…
ƻ

Event
@TransactionalEventListener
…
@TransactionalEventListener
…
Transaction Commit
31
@TransactionalEventListener
@TransactionalEventListener
…
ƻ

Event
@TransactionalEventListener
…
@TransactionalEventListener
…
Transaction Commit
31
@TransactionalEventListener
@TransactionalEventListener
…
ƻ

Event
@TransactionalEventListener
…
@TransactionalEventListener
…
Transaction Commit
31
@TransactionalEventListener
@TransactionalEventListener
…
ƻ

Event
@TransactionalEventListener
…
@TransactionalEventListener
…
Transaction Commit
31
@TransactionalEventListener
@TransactionalEventListener
…
ƻ

Event
@TransactionalEventListener
…
@TransactionalEventListener
…
Transaction Commit
31
@TransactionalEventListener
@TransactionalEventListener
…
ƻ

Event
@TransactionalEventListener
…
@TransactionalEventListener
…
Transaction Commit
Event Publication Registry
1. Write application event publication log for transactional listeners
On application event publication a log entry is written for every event and transactional event listener
interested in it. That way, the transaction remembers which events have to be properly handled and in
case listener invocations fail or the application crashes events can be re-published.
2. Transaction listeners are decorated to register successful completion
Transactional event listeners are decorated with an interceptor that marks the log entry for the listener
invocation on successful listener completion. When all listeners were handled, the log only contains
publication logs for the ones that failed.
3. Incomplete publications can be retried
Either periodically or during application restarts.
32
Event publication registry
DEMO
33
architectureComponents & relationships
36
Orders
37
Inventory
Component Dependency Type dependency
38
Orders
Inventory
Component Dependency Type dependency


Event
Event publication
39
Imperative style
1. Spring bean invocations across Bounded Contexts
The OrderManagement actively invokes other Spring Beans to trigger peripheral functionality. By that,
it becomes a spot of gravity that is likely to become more complex over time and pollute the
transaction (i.e. it's doing too much). There's no explicit representation of the business incident
"completion of an order" except the method name.
2. Testing focussed on explicit interaction
Unit testing usually works with mocks and requires complex setup of expected behavior and
verifications. For integration tests, dependency beans need to be available, too, which makes it hard to
test the module in isolation.
3. Post-commit functionality hard to integrate
As the entire method is running inside a transaction, it becomes harder to integrate logic that is
supposed to run after the transaction has committed, for example sending a confirmation email.
Especially interactions with external resources that can take a while should be held outside the main
transaction.
40
Event based implementation
1. Less knowledge in the order module and explicit integration point
Introducing the OrderCompleted event allows the order module to get rid of the knowledge about
which peripheral functionality to integrate and thus allows for easier extension. Whether that
downstream functionality is executed synchronously as part of the main transaction, asynchronously or
after the transaction is now determined by the event listener.
2. Testing shifts to verification of events published
The reduced number of dependencies of a component significantly eases testing as these dependencies
don't have to be mocked and prepared (in the case of a unit test) nor do they have to be available as
Spring bean in an integration test. In the former case, Modulith's PublishedEvents API allows to write
readable assertions on the events that were produced during the test execution. The latter enables
Modulith's support for @ModuleTest to integration test a single module.
41
Resources
44
Sample code
Project website on GitHub
Moduliths
Spring Boot extension to support building modular monoliths, incl.
support for events, testing etc., Project website on GitHub
Spring Domain Events
Prototype implementation of an event publication registry, likely to
be integrated into Moduliths, Project website on GitHub
Refactoring to a System of Systems
Video on YouTube
Thank you!
45
Oliver Drotbohm ƀ odrotbohm@vmware.com/ odrotbohm

More Related Content

What's hot

Advanced Streaming Analytics with Apache Flink and Apache Kafka, Stephan Ewen
Advanced Streaming Analytics with Apache Flink and Apache Kafka, Stephan EwenAdvanced Streaming Analytics with Apache Flink and Apache Kafka, Stephan Ewen
Advanced Streaming Analytics with Apache Flink and Apache Kafka, Stephan Ewen
confluent
 
Designing ETL Pipelines with Structured Streaming and Delta Lake—How to Archi...
Designing ETL Pipelines with Structured Streaming and Delta Lake—How to Archi...Designing ETL Pipelines with Structured Streaming and Delta Lake—How to Archi...
Designing ETL Pipelines with Structured Streaming and Delta Lake—How to Archi...
Databricks
 

What's hot (20)

Distributed Transaction in Microservice
Distributed Transaction in MicroserviceDistributed Transaction in Microservice
Distributed Transaction in Microservice
 
Stream All Things—Patterns of Modern Data Integration with Gwen Shapira
Stream All Things—Patterns of Modern Data Integration with Gwen ShapiraStream All Things—Patterns of Modern Data Integration with Gwen Shapira
Stream All Things—Patterns of Modern Data Integration with Gwen Shapira
 
API
APIAPI
API
 
Distributed load testing with k6
Distributed load testing with k6Distributed load testing with k6
Distributed load testing with k6
 
API : l'architecture REST
API : l'architecture RESTAPI : l'architecture REST
API : l'architecture REST
 
Microservice With Spring Boot and Spring Cloud
Microservice With Spring Boot and Spring CloudMicroservice With Spring Boot and Spring Cloud
Microservice With Spring Boot and Spring Cloud
 
Service discovery with Eureka and Spring Cloud
Service discovery with Eureka and Spring CloudService discovery with Eureka and Spring Cloud
Service discovery with Eureka and Spring Cloud
 
Introduction to Kafka connect
Introduction to Kafka connectIntroduction to Kafka connect
Introduction to Kafka connect
 
Running distributed tests with k6.pdf
Running distributed tests with k6.pdfRunning distributed tests with k6.pdf
Running distributed tests with k6.pdf
 
Processing Semantically-Ordered Streams in Financial Services
Processing Semantically-Ordered Streams in Financial ServicesProcessing Semantically-Ordered Streams in Financial Services
Processing Semantically-Ordered Streams in Financial Services
 
Event-driven Architecture
Event-driven ArchitectureEvent-driven Architecture
Event-driven Architecture
 
Advanced Streaming Analytics with Apache Flink and Apache Kafka, Stephan Ewen
Advanced Streaming Analytics with Apache Flink and Apache Kafka, Stephan EwenAdvanced Streaming Analytics with Apache Flink and Apache Kafka, Stephan Ewen
Advanced Streaming Analytics with Apache Flink and Apache Kafka, Stephan Ewen
 
Apache Kafka Architecture & Fundamentals Explained
Apache Kafka Architecture & Fundamentals ExplainedApache Kafka Architecture & Fundamentals Explained
Apache Kafka Architecture & Fundamentals Explained
 
Sapo Microservices Architecture
Sapo Microservices ArchitectureSapo Microservices Architecture
Sapo Microservices Architecture
 
Event Driven Microservices architecture
Event Driven Microservices architectureEvent Driven Microservices architecture
Event Driven Microservices architecture
 
REST API and CRUD
REST API and CRUDREST API and CRUD
REST API and CRUD
 
Designing ETL Pipelines with Structured Streaming and Delta Lake—How to Archi...
Designing ETL Pipelines with Structured Streaming and Delta Lake—How to Archi...Designing ETL Pipelines with Structured Streaming and Delta Lake—How to Archi...
Designing ETL Pipelines with Structured Streaming and Delta Lake—How to Archi...
 
Apache Nifi Crash Course
Apache Nifi Crash CourseApache Nifi Crash Course
Apache Nifi Crash Course
 
Spring Framework
Spring FrameworkSpring Framework
Spring Framework
 
Introduction to Spring Framework
Introduction to Spring FrameworkIntroduction to Spring Framework
Introduction to Spring Framework
 

Similar to A Deep Dive into Spring Application Events

Restate: Event-driven Asynchronous Services, Easy as Synchronous RPC
Restate: Event-driven Asynchronous Services, Easy as Synchronous RPCRestate: Event-driven Asynchronous Services, Easy as Synchronous RPC
Restate: Event-driven Asynchronous Services, Easy as Synchronous RPC
HostedbyConfluent
 
Swiss army knife Spring
Swiss army knife SpringSwiss army knife Spring
Swiss army knife Spring
Mario Fusco
 
FIWARE CEP GE introduction, ICT 2015
FIWARE CEP GE introduction, ICT 2015FIWARE CEP GE introduction, ICT 2015
FIWARE CEP GE introduction, ICT 2015
ishkin
 

Similar to A Deep Dive into Spring Application Events (20)

Restate: Event-driven Asynchronous Services, Easy as Synchronous RPC
Restate: Event-driven Asynchronous Services, Easy as Synchronous RPCRestate: Event-driven Asynchronous Services, Easy as Synchronous RPC
Restate: Event-driven Asynchronous Services, Easy as Synchronous RPC
 
Swiss army knife Spring
Swiss army knife SpringSwiss army knife Spring
Swiss army knife Spring
 
Events: The Object Oriented Hook System.
Events: The Object Oriented Hook System.Events: The Object Oriented Hook System.
Events: The Object Oriented Hook System.
 
Vldb 2010 event processing tutorial
Vldb 2010 event processing tutorialVldb 2010 event processing tutorial
Vldb 2010 event processing tutorial
 
ThingMonk 2016 - Concursus Event sourcing for the IOT By Tareq Abedrabbo & Do...
ThingMonk 2016 - Concursus Event sourcing for the IOT By Tareq Abedrabbo & Do...ThingMonk 2016 - Concursus Event sourcing for the IOT By Tareq Abedrabbo & Do...
ThingMonk 2016 - Concursus Event sourcing for the IOT By Tareq Abedrabbo & Do...
 
Aaai 2011 event processing tutorial
Aaai 2011 event processing tutorialAaai 2011 event processing tutorial
Aaai 2011 event processing tutorial
 
Event Driven Architecture
Event Driven ArchitectureEvent Driven Architecture
Event Driven Architecture
 
PATTERNS06 - The .NET Event Model
PATTERNS06 - The .NET Event ModelPATTERNS06 - The .NET Event Model
PATTERNS06 - The .NET Event Model
 
In the Eventual Consistency of Succeeding at Microservices
In the Eventual Consistency of Succeeding at MicroservicesIn the Eventual Consistency of Succeeding at Microservices
In the Eventual Consistency of Succeeding at Microservices
 
Axon Framework, Exploring CQRS and Event Sourcing Architecture
Axon Framework, Exploring CQRS and Event Sourcing ArchitectureAxon Framework, Exploring CQRS and Event Sourcing Architecture
Axon Framework, Exploring CQRS and Event Sourcing Architecture
 
Data Microservices with Spring Cloud
Data Microservices with Spring CloudData Microservices with Spring Cloud
Data Microservices with Spring Cloud
 
FIWARE CEP GE introduction, ICT 2015
FIWARE CEP GE introduction, ICT 2015FIWARE CEP GE introduction, ICT 2015
FIWARE CEP GE introduction, ICT 2015
 
IoT in salsa Serverless
IoT in salsa ServerlessIoT in salsa Serverless
IoT in salsa Serverless
 
PROGRAMMING IN JAVA- unit 4-part II
PROGRAMMING IN JAVA- unit 4-part IIPROGRAMMING IN JAVA- unit 4-part II
PROGRAMMING IN JAVA- unit 4-part II
 
Platform Events by Tim Taylor
Platform Events by Tim TaylorPlatform Events by Tim Taylor
Platform Events by Tim Taylor
 
Eda on the azure services platform
Eda on the azure services platformEda on the azure services platform
Eda on the azure services platform
 
Fabric - Realtime stream processing framework
Fabric - Realtime stream processing frameworkFabric - Realtime stream processing framework
Fabric - Realtime stream processing framework
 
Complex Event Processing
Complex Event ProcessingComplex Event Processing
Complex Event Processing
 
The Significant role of event driven apps in software development
The Significant role of event driven apps in software development					The Significant role of event driven apps in software development
The Significant role of event driven apps in software development
 
Put Events to Work and Respond in Real Time
Put Events to Work and Respond in Real TimePut Events to Work and Respond in Real Time
Put Events to Work and Respond in Real Time
 

More from VMware Tanzu

More from VMware Tanzu (20)

Spring into AI presented by Dan Vega 5/14
Spring into AI presented by Dan Vega 5/14Spring into AI presented by Dan Vega 5/14
Spring into AI presented by Dan Vega 5/14
 
What AI Means For Your Product Strategy And What To Do About It
What AI Means For Your Product Strategy And What To Do About ItWhat AI Means For Your Product Strategy And What To Do About It
What AI Means For Your Product Strategy And What To Do About It
 
Make the Right Thing the Obvious Thing at Cardinal Health 2023
Make the Right Thing the Obvious Thing at Cardinal Health 2023Make the Right Thing the Obvious Thing at Cardinal Health 2023
Make the Right Thing the Obvious Thing at Cardinal Health 2023
 
Enhancing DevEx and Simplifying Operations at Scale
Enhancing DevEx and Simplifying Operations at ScaleEnhancing DevEx and Simplifying Operations at Scale
Enhancing DevEx and Simplifying Operations at Scale
 
Spring Update | July 2023
Spring Update | July 2023Spring Update | July 2023
Spring Update | July 2023
 
Platforms, Platform Engineering, & Platform as a Product
Platforms, Platform Engineering, & Platform as a ProductPlatforms, Platform Engineering, & Platform as a Product
Platforms, Platform Engineering, & Platform as a Product
 
Building Cloud Ready Apps
Building Cloud Ready AppsBuilding Cloud Ready Apps
Building Cloud Ready Apps
 
Spring Boot 3 And Beyond
Spring Boot 3 And BeyondSpring Boot 3 And Beyond
Spring Boot 3 And Beyond
 
Spring Cloud Gateway - SpringOne Tour 2023 Charles Schwab.pdf
Spring Cloud Gateway - SpringOne Tour 2023 Charles Schwab.pdfSpring Cloud Gateway - SpringOne Tour 2023 Charles Schwab.pdf
Spring Cloud Gateway - SpringOne Tour 2023 Charles Schwab.pdf
 
Simplify and Scale Enterprise Apps in the Cloud | Boston 2023
Simplify and Scale Enterprise Apps in the Cloud | Boston 2023Simplify and Scale Enterprise Apps in the Cloud | Boston 2023
Simplify and Scale Enterprise Apps in the Cloud | Boston 2023
 
Simplify and Scale Enterprise Apps in the Cloud | Seattle 2023
Simplify and Scale Enterprise Apps in the Cloud | Seattle 2023Simplify and Scale Enterprise Apps in the Cloud | Seattle 2023
Simplify and Scale Enterprise Apps in the Cloud | Seattle 2023
 
tanzu_developer_connect.pptx
tanzu_developer_connect.pptxtanzu_developer_connect.pptx
tanzu_developer_connect.pptx
 
Tanzu Virtual Developer Connect Workshop - French
Tanzu Virtual Developer Connect Workshop - FrenchTanzu Virtual Developer Connect Workshop - French
Tanzu Virtual Developer Connect Workshop - French
 
Tanzu Developer Connect Workshop - English
Tanzu Developer Connect Workshop - EnglishTanzu Developer Connect Workshop - English
Tanzu Developer Connect Workshop - English
 
Virtual Developer Connect Workshop - English
Virtual Developer Connect Workshop - EnglishVirtual Developer Connect Workshop - English
Virtual Developer Connect Workshop - English
 
Tanzu Developer Connect - French
Tanzu Developer Connect - FrenchTanzu Developer Connect - French
Tanzu Developer Connect - French
 
Simplify and Scale Enterprise Apps in the Cloud | Dallas 2023
Simplify and Scale Enterprise Apps in the Cloud | Dallas 2023Simplify and Scale Enterprise Apps in the Cloud | Dallas 2023
Simplify and Scale Enterprise Apps in the Cloud | Dallas 2023
 
SpringOne Tour: Deliver 15-Factor Applications on Kubernetes with Spring Boot
SpringOne Tour: Deliver 15-Factor Applications on Kubernetes with Spring BootSpringOne Tour: Deliver 15-Factor Applications on Kubernetes with Spring Boot
SpringOne Tour: Deliver 15-Factor Applications on Kubernetes with Spring Boot
 
SpringOne Tour: The Influential Software Engineer
SpringOne Tour: The Influential Software EngineerSpringOne Tour: The Influential Software Engineer
SpringOne Tour: The Influential Software Engineer
 
SpringOne Tour: Domain-Driven Design: Theory vs Practice
SpringOne Tour: Domain-Driven Design: Theory vs PracticeSpringOne Tour: Domain-Driven Design: Theory vs Practice
SpringOne Tour: Domain-Driven Design: Theory vs Practice
 

Recently uploaded

Abortion Pills For Sale WhatsApp[[+27737758557]] In Birch Acres, Abortion Pil...
Abortion Pills For Sale WhatsApp[[+27737758557]] In Birch Acres, Abortion Pil...Abortion Pills For Sale WhatsApp[[+27737758557]] In Birch Acres, Abortion Pil...
Abortion Pills For Sale WhatsApp[[+27737758557]] In Birch Acres, Abortion Pil...
drm1699
 

Recently uploaded (20)

Novo Nordisk: When Knowledge Graphs meet LLMs
Novo Nordisk: When Knowledge Graphs meet LLMsNovo Nordisk: When Knowledge Graphs meet LLMs
Novo Nordisk: When Knowledge Graphs meet LLMs
 
Encryption Recap: A Refresher on Key Concepts
Encryption Recap: A Refresher on Key ConceptsEncryption Recap: A Refresher on Key Concepts
Encryption Recap: A Refresher on Key Concepts
 
Abortion Pills For Sale WhatsApp[[+27737758557]] In Birch Acres, Abortion Pil...
Abortion Pills For Sale WhatsApp[[+27737758557]] In Birch Acres, Abortion Pil...Abortion Pills For Sale WhatsApp[[+27737758557]] In Birch Acres, Abortion Pil...
Abortion Pills For Sale WhatsApp[[+27737758557]] In Birch Acres, Abortion Pil...
 
Abortion Clinic In Pongola ](+27832195400*)[ 🏥 Safe Abortion Pills In Pongola...
Abortion Clinic In Pongola ](+27832195400*)[ 🏥 Safe Abortion Pills In Pongola...Abortion Clinic In Pongola ](+27832195400*)[ 🏥 Safe Abortion Pills In Pongola...
Abortion Clinic In Pongola ](+27832195400*)[ 🏥 Safe Abortion Pills In Pongola...
 
Wired_2.0_CREATE YOUR ULTIMATE LEARNING ENVIRONMENT_JCON_16052024
Wired_2.0_CREATE YOUR ULTIMATE LEARNING ENVIRONMENT_JCON_16052024Wired_2.0_CREATE YOUR ULTIMATE LEARNING ENVIRONMENT_JCON_16052024
Wired_2.0_CREATE YOUR ULTIMATE LEARNING ENVIRONMENT_JCON_16052024
 
Abortion Clinic In Johannesburg ](+27832195400*)[ 🏥 Safe Abortion Pills in Jo...
Abortion Clinic In Johannesburg ](+27832195400*)[ 🏥 Safe Abortion Pills in Jo...Abortion Clinic In Johannesburg ](+27832195400*)[ 🏥 Safe Abortion Pills in Jo...
Abortion Clinic In Johannesburg ](+27832195400*)[ 🏥 Safe Abortion Pills in Jo...
 
Navigation in flutter – how to add stack, tab, and drawer navigators to your ...
Navigation in flutter – how to add stack, tab, and drawer navigators to your ...Navigation in flutter – how to add stack, tab, and drawer navigators to your ...
Navigation in flutter – how to add stack, tab, and drawer navigators to your ...
 
Abortion Pill Prices Jane Furse ](+27832195400*)[ 🏥 Women's Abortion Clinic i...
Abortion Pill Prices Jane Furse ](+27832195400*)[ 🏥 Women's Abortion Clinic i...Abortion Pill Prices Jane Furse ](+27832195400*)[ 🏥 Women's Abortion Clinic i...
Abortion Pill Prices Jane Furse ](+27832195400*)[ 🏥 Women's Abortion Clinic i...
 
微信号购买
微信号购买微信号购买
微信号购买
 
The mythical technical debt. (Brooke, please, forgive me)
The mythical technical debt. (Brooke, please, forgive me)The mythical technical debt. (Brooke, please, forgive me)
The mythical technical debt. (Brooke, please, forgive me)
 
GraphSummit Milan - Neo4j: The Art of the Possible with Graph
GraphSummit Milan - Neo4j: The Art of the Possible with GraphGraphSummit Milan - Neo4j: The Art of the Possible with Graph
GraphSummit Milan - Neo4j: The Art of the Possible with Graph
 
What Goes Wrong with Language Definitions and How to Improve the Situation
What Goes Wrong with Language Definitions and How to Improve the SituationWhat Goes Wrong with Language Definitions and How to Improve the Situation
What Goes Wrong with Language Definitions and How to Improve the Situation
 
Lessons Learned from Building a Serverless Notifications System.pdf
Lessons Learned from Building a Serverless Notifications System.pdfLessons Learned from Building a Serverless Notifications System.pdf
Lessons Learned from Building a Serverless Notifications System.pdf
 
Abortion Pill Prices Mthatha (@](+27832195400*)[ 🏥 Women's Abortion Clinic In...
Abortion Pill Prices Mthatha (@](+27832195400*)[ 🏥 Women's Abortion Clinic In...Abortion Pill Prices Mthatha (@](+27832195400*)[ 🏥 Women's Abortion Clinic In...
Abortion Pill Prices Mthatha (@](+27832195400*)[ 🏥 Women's Abortion Clinic In...
 
Effective Strategies for Wix's Scaling challenges - GeeCon
Effective Strategies for Wix's Scaling challenges - GeeConEffective Strategies for Wix's Scaling challenges - GeeCon
Effective Strategies for Wix's Scaling challenges - GeeCon
 
Abortion Clinic In Pretoria ](+27832195400*)[ 🏥 Safe Abortion Pills in Pretor...
Abortion Clinic In Pretoria ](+27832195400*)[ 🏥 Safe Abortion Pills in Pretor...Abortion Clinic In Pretoria ](+27832195400*)[ 🏥 Safe Abortion Pills in Pretor...
Abortion Clinic In Pretoria ](+27832195400*)[ 🏥 Safe Abortion Pills in Pretor...
 
Abortion Pill Prices Aliwal North ](+27832195400*)[ 🏥 Women's Abortion Clinic...
Abortion Pill Prices Aliwal North ](+27832195400*)[ 🏥 Women's Abortion Clinic...Abortion Pill Prices Aliwal North ](+27832195400*)[ 🏥 Women's Abortion Clinic...
Abortion Pill Prices Aliwal North ](+27832195400*)[ 🏥 Women's Abortion Clinic...
 
Wired_2.0_CREATE YOUR ULTIMATE LEARNING ENVIRONMENT_JCON_16052024
Wired_2.0_CREATE YOUR ULTIMATE LEARNING ENVIRONMENT_JCON_16052024Wired_2.0_CREATE YOUR ULTIMATE LEARNING ENVIRONMENT_JCON_16052024
Wired_2.0_CREATE YOUR ULTIMATE LEARNING ENVIRONMENT_JCON_16052024
 
Anypoint Code Builder - Munich MuleSoft Meetup - 16th May 2024
Anypoint Code Builder - Munich MuleSoft Meetup - 16th May 2024Anypoint Code Builder - Munich MuleSoft Meetup - 16th May 2024
Anypoint Code Builder - Munich MuleSoft Meetup - 16th May 2024
 
Alluxio Monthly Webinar | Simplify Data Access for AI in Multi-Cloud
Alluxio Monthly Webinar | Simplify Data Access for AI in Multi-CloudAlluxio Monthly Webinar | Simplify Data Access for AI in Multi-Cloud
Alluxio Monthly Webinar | Simplify Data Access for AI in Multi-Cloud
 

A Deep Dive into Spring Application Events