Categories: Camel프레임워크

[Camel in Action] 11-1. Camel 에러 핸들링 완전 정복 – onException과 DeadLetterChannel

Camel에서 에러 핸들링이 중요한 이유

외부 시스템과 연동하는 통합 애플리케이션에서 에러는 예외가 아니라 일상입니다. 네트워크 장애, 외부 API 다운, 잘못된 데이터 형식 등 다양한 이유로 처리 중 예외가 발생합니다. Camel은 이런 상황을 체계적으로 처리하기 위한 강력한 에러 핸들링 메커니즘을 제공합니다.

기본 에러 전파 방식

Camel 라우트에서 예외가 발생하면 기본적으로 호출자에게 예외가 전파됩니다. Consumer(소비자)에서 발생한 경우 예외 처리 방식은 컴포넌트마다 다릅니다.

  • JMS: 메시지가 롤백되어 재시도 큐로 이동
  • File: 파일을 moveFailed 디렉터리로 이동
  • HTTP: 에러 응답 코드 반환

onException으로 특정 예외 처리

특정 예외 타입에 따라 다른 처리를 하려면 onException을 사용합니다. RouteBuilder의 configure() 메서드 상단에 선언합니다.

public class OrderRoutes extends RouteBuilder {
  @Override
  public void configure() throws Exception {

    // 유효성 검사 실패: 에러 응답 반환
    onException(ValidationException.class)
      .handled(true)
      .setHeader(Exchange.HTTP_RESPONSE_CODE, constant(400))
      .setBody(simple("잘못된 요청: ${exception.message}"));

    // 외부 서비스 연결 실패: 3번 재시도 후 DLQ로
    onException(ConnectException.class)
      .maximumRedeliveries(3)
      .redeliveryDelay(2000)
      .backOffMultiplier(2) // 지수 백오프: 2초, 4초, 8초
      .useExponentialBackOff()
      .to("activemq:queue:failed-orders");

    from("direct:processOrder")
      .to("bean:orderService");
  }
}

handled(true)는 예외를 소비해서 더 이상 전파되지 않게 합니다. 없으면 처리 후에도 예외가 계속 위로 전파됩니다.

Dead Letter Channel – 실패 메시지 격리

Dead Letter Channel은 처리 실패 메시지를 특정 엔드포인트(Dead Letter Queue)로 보내는 에러 핸들러입니다.

errorHandler(deadLetterChannel("activemq:queue:DLQ")
  .maximumRedeliveries(3)
  .redeliveryDelay(1000)
  .retryAttemptedLogLevel(LoggingLevel.WARN)
  .logStackTrace(true));

DLQ를 모니터링하면 처리 실패 메시지를 나중에 수동으로 재처리하거나 원인을 분석할 수 있습니다.

Retry와 지수 백오프 전략

일시적인 장애에는 즉시 재시도보다 점진적으로 간격을 늘리는 지수 백오프가 효과적입니다. 상대방 서버에 부하를 주지 않으면서 복구 시간을 줍니다.

onException(IOException.class)
  .maximumRedeliveries(5)
  .redeliveryDelay(1000)       // 첫 재시도: 1초 후
  .backOffMultiplier(2.0)       // 다음: 2초, 4초, 8초, 16초
  .useExponentialBackOff()
  .maximumRedeliveryDelay(60000) // 최대 60초 간격
  .logRetryAttempted(true);

예외 정보 활용

예외 처리 중에 예외 정보를 메시지에 담아 DLQ나 알림으로 보낼 수 있습니다.

onException(Exception.class)
  .handled(true)
  .process(exchange -> {
    Exception cause = exchange.getProperty(Exchange.EXCEPTION_CAUGHT, Exception.class);
    exchange.getIn().setHeader("errorMessage", cause.getMessage());
    exchange.getIn().setHeader("errorClass", cause.getClass().getName());
  })
  .to("activemq:queue:error-notifications");

zerg96

Recent Posts

[Apache Camel] 2025년 최신 트렌드 – AI 통합과 서버리스 Camel의 미래

2025년 Apache Camel의 최신 트렌드를 분석합니다. AI/LLM 통합 컴포넌트, 서버리스 배포, Camel K 진화, WebAssembly…

10시간 ago

[Camel in Action] 완결편 – Apache Camel 전체 여정 회고와 다음 단계

Camel in Action을 완독한 후 Apache Camel의 전체 그림을 다시 정리합니다. 핵심 철학, 학습 경로,…

10시간 ago

[Camel in Action] 실전편 – Camel 마이그레이션 가이드 2.x에서 4.x까지

Apache Camel 2.x에서 3.x, 4.x로 마이그레이션하는 단계별 가이드입니다. 주요 API 변경사항, 제거된 컴포넌트, 자동화 도구…

10시간 ago

[Camel in Action] 실전편 – Camel 라우트 디버깅 기법과 문제 해결 가이드

Apache Camel 라우트에서 발생하는 문제를 디버깅하고 해결하는 실전 기법을 설명합니다. 로그 분석, breakpoint 디버깅, Tracer,…

10시간 ago

[Camel in Action] 실전편 – Camel 도입 전 반드시 알아야 할 것들

Apache Camel을 프로젝트에 도입하기 전 알아야 할 핵심 사항을 정리합니다. 학습 곡선, 도입 비용, 적합한…

10시간 ago

[Camel in Action] 실전편 – Enterprise Integration Patterns 20가지 핵심 정리

엔터프라이즈 통합 패턴(EIP) 20가지를 Apache Camel 코드와 함께 한 번에 정리합니다. 메시징 채널, 메시지 라우팅,…

10시간 ago