spring-events

SKILL.md

Spring Application Events

Quick Start

// Custom Event (POJO - preferred)
public record OrderCreatedEvent(
    Long orderId,
    Long customerId,
    BigDecimal totalAmount,
    Instant createdAt
) {}

// Publisher
@Service
@RequiredArgsConstructor
public class OrderService {

    private final ApplicationEventPublisher eventPublisher;

    @Transactional
    public Order createOrder(CreateOrderRequest request) {
        Order order = orderRepository.save(new Order(request));
        eventPublisher.publishEvent(new OrderCreatedEvent(
            order.getId(), order.getCustomerId(),
            order.getTotalAmount(), order.getCreatedAt()
        ));
        return order;
    }
}

// Listener
@Component
@Slf4j
public class OrderEventListener {

    @EventListener
    public void handleOrderCreated(OrderCreatedEvent event) {
        log.info("Order created: {}", event.orderId());
    }
}

@EventListener

@Component
public class EventListeners {

    // Basic listener
    @EventListener
    public void handleOrderCreated(OrderCreatedEvent event) {
        log.info("Processing order: {}", event.orderId());
    }

    // Conditional listener
    @EventListener(condition = "#event.totalAmount > 1000")
    public void handleLargeOrder(OrderCreatedEvent event) {
        notifyManager(event);
    }

    // Ordered execution
    @EventListener
    @Order(1)  // Executed first
    public void validateOrder(OrderCreatedEvent event) { }

    @EventListener
    @Order(2)  // Executed second
    public void processOrder(OrderCreatedEvent event) { }

    // Multiple event types
    @EventListener({OrderCreatedEvent.class, OrderUpdatedEvent.class})
    public void handleOrderChange(Object event) { }
}

Event Chain (Publish New Event from Listener)

@EventListener
public NotificationEvent handleOrderCreated(OrderCreatedEvent event) {
    return new NotificationEvent(event.customerId(), "Order created!");
}

@EventListener
public Collection<Object> handleOrderShipped(OrderShippedEvent event) {
    return List.of(
        new NotificationEvent(event.customerId(), "Order shipped!"),
        new AnalyticsEvent("order_shipped", event.orderId())
    );
}

Custom Events

// Generic event
public class EntityEvent<T> {
    private final T entity;
    private final EventType type;
    private final Instant timestamp = Instant.now();

    public enum EventType { CREATED, UPDATED, DELETED }
}

// Generic publisher
@Component
public class EntityEventPublisher {
    private final ApplicationEventPublisher publisher;

    public <T> void publishCreated(T entity) {
        publisher.publishEvent(new EntityEvent<>(entity, EventType.CREATED));
    }
}

// Typed listener
@EventListener
public void handleUserEvent(EntityEvent<User> event) {
    switch (event.getType()) {
        case CREATED -> handleUserCreated(event.getEntity());
        case UPDATED -> handleUserUpdated(event.getEntity());
    }
}

Full Reference: See transactional.md for @TransactionalEventListener, Async Events.


@TransactionalEventListener

// Execute AFTER transaction commit (default)
@TransactionalEventListener
public void handleAfterCommit(OrderCreatedEvent event) {
    emailService.sendOrderConfirmation(event.orderId());
}

// Execute AFTER rollback
@TransactionalEventListener(phase = TransactionPhase.AFTER_ROLLBACK)
public void handleAfterRollback(OrderCreatedEvent event) {
    alertService.notifyRollback(event);
}

// Execute BEFORE commit
@TransactionalEventListener(phase = TransactionPhase.BEFORE_COMMIT)
public void handleBeforeCommit(OrderCreatedEvent event) {
    validateOrderBeforeCommit(event);
}

// Fallback if no transaction
@TransactionalEventListener(fallbackExecution = true)
public void handleWithFallback(OrderCreatedEvent event) { }

Full Reference: See patterns.md for Domain Events, Aggregate Root, Event Store.


Best Practices

  • ✅ Use @TransactionalEventListener for side effects
  • ✅ Use @Async for non-critical operations
  • ✅ Implement retry for fallible listeners
  • ✅ Use immutable events (records)
  • ✅ Define order with @Order if needed
  • ❌ Don't modify state in sync listeners
  • ❌ Don't assume execution order without @Order

Production Checklist

  • Event classes immutabili
  • TransactionalEventListener for external calls
  • Async for non-critical operations
  • Error handling implemented
  • Retry for transient operations
  • Monitoring events

When NOT to Use This Skill

  • Distributed events - Use spring-kafka or spring-amqp
  • Guaranteed delivery - Use messaging systems
  • Event sourcing - Consider Axon Framework

Common Pitfalls

Error Cause Solution
Listener not executed Missing @Component Add annotation
Event lost on rollback Using @EventListener Use @TransactionalEventListener
Deadlock Sync listener calls same service Use @Async
Exception hidden Async void Implement error handler

Anti-Patterns

Anti-Pattern Problem Solution
Sync events in transaction Long transactions Use @Async or @TransactionalEventListener
Circular event publishing Infinite loop Guard with flags
Heavy processing in sync Blocks publisher Use async listeners
Modifying event after publish Shared state issues Make events immutable

Quick Troubleshooting

Problem Diagnostic Fix
Listener not invoked Check @EventListener Verify component scanned
Transaction not committed Check event phase Use AFTER_COMMIT
Async not working Check @EnableAsync Add to config
Order matters Listeners random Use @Order

Reference Files

File Content
transactional.md @TransactionalEventListener, Async Events, Lifecycle
patterns.md Domain Events, Aggregate Root, Event Store

External Documentation

Weekly Installs
10
GitHub Stars
2
First Seen
10 days ago
Installed on
cursor9
gemini-cli9
amp9
cline9
github-copilot9
codex9