데이터베이스 트리거가 종종 문제를 일으키는 이유
Takashi Yamamoto
Infrastructure Engineer · Leapcell

소개
소프트웨어 개발 세계에서 데이터베이스는 종종 애플리케이션의 심장 역할을 하며 중요한 정보를 저장하고 관리합니다. 개발자들은 데이터 무결성을 강제하고 특정 작업을 자동화할 방법을 자주 찾습니다. 이를 달성하기 위한 일반적인 메커니즘 중 하나가 데이터베이스 트리거입니다. 트리거는 처음에는 데이터 수정에 반응하고 데이터베이스 내에서 미리 정의된 작업을 직접 실행할 수 있는 강력한 도구처럼 보입니다. 이는 비즈니스 규칙을 데이터 자체에 가깝게 내장하여 개발을 단순화할 것을 약속합니다. 그러나 이 편리해 보이는 접근 방식은 곧 테스트, 디버깅 및 전반적인 시스템 유지보수를 복잡하게 만드는 수많은 문제로 이어질 수 있습니다. 이 기사에서는 데이터베이스 트리거에 크게 의존하는 것의 함정을 탐구하고 비즈니스 로직을 관리하기 위한 더 강력하고 유연한 접근 방식을 옹호할 것입니다.
데이터베이스 트리거의 함정과 비즈니스 로직이 진정으로 속한 곳
트리거를 피해야 하는 이유를 살펴보기 전에 몇 가지 핵심 용어를 간략하게 정의해 보겠습니다.
데이터베이스 트리거: 데이터베이스의 특정 테이블 또는 뷰에 대한 특정 이벤트에 응답하여 자동으로 실행되는 저장 절차 코드입니다. 이러한 이벤트에는 INSERT, UPDATE, DELETE 작업이 포함될 수 있습니다.
비즈니스 로직: 데이터베이스와 사용자 인터페이스 간 또는 시스템의 다른 부분 간의 정보 교환을 처리하는 사용자 지정 규칙 또는 알고리즘입니다. 데이터가 생성, 저장, 변경되는 방식과 시스템 전체와 어떻게 상호 작용하는지를 정의합니다.
데이터 무결성: 전체 수명 주기 동안 데이터의 정확성과 일관성.
트리거의 숨겨진 비용
트리거는 데이터 무결성을 강제하고 작업을 자동화할 수 있지만, 그 기능에는 상당한 단점이 따릅니다.
- 
디버깅 저하: 트리거를 디버깅하는 것은 악명 높게 어렵습니다. IDE에서 한 줄씩 실행할 수 있는 애플리케이션 코드와 달리 트리거 실행은 종종 불투명합니다. 오류는 실제 트리거 호출에서 멀리 떨어진 미묘한 방식으로 나타날 수 있어 근본 원인을 파악하기 어렵습니다. 여러 트리거가 함께 연결되면 이 복잡성이 기하급수적으로 증가하여 암묵적인 종속성의 복잡한 네트워크가 형성됩니다.
 - 
유지보수성 저하: 트리거에 내장된 비즈니스 로직은 종종 SQL 또는 데이터베이스별 절차 언어(PL/SQL 또는 T-SQL 등)로 작성됩니다. 이 코드는 일반적으로 애플리케이션 수준 코드보다 읽고, 이해하고, 수정하기가 더 어렵습니다. 비즈니스 요구 사항이 발전함에 따라 트리거 로직을 변경하는 것은 새로운 버그나 의도하지 않은 부작용을 도입할 가능성이 매우 높은 고통스러운 프로세스가 될 수 있습니다. 또한 비즈니스 규칙을 데이터베이스 스키마에 단단히 결합하여 스키마 변경을 더 복잡하게 만듭니다.
 - 
성능 오버헤드: 트리거는 트리거를 발생시키는 데이터베이스 작업과 동기적으로 실행됩니다. 트리거가 복잡한 계산을 수행하거나, 외부 함수를 호출하거나, 다른 테이블과 상호 작용하는 경우
INSERT,UPDATE,DELETE작업을 크게 늦출 수 있습니다. 이 오버헤드는 특히 고거래량 환경에서 문제가 될 수 있으며, 병목 현상과 애플리케이션 성능 저하로 이어집니다. - 
테스트 어려움: 데이터베이스 트리거의 단위 테스트는 애플리케이션 코드의 단위 테스트보다 훨씬 복잡합니다. 특정 데이터베이스 상태를 설정하고, DML 작업을 실행한 다음, 결과 데이터베이스 상태를 어설션해야 하는 경우가 많습니다. 이는 테스트를 데이터베이스에 단단히 결합하여 테스트를 더 느리고, 분리하기 어렵고, 더 취약하게 만듭니다.
 - 
애플리케이션 제어 상실: 비즈니스 로직이 트리거에 상주하면 애플리케이션은 이러한 작업에 대한 직접적인 제어 및 가시성을 잃게 됩니다. 애플리케이션에서 실행된
UPDATE문은 기본 트리거로 인해 예상치 못한 결과를 초래할 수 있으며, 이는 애플리케이션 계층에서 예측하거나 관리하기 어려운 암묵적인 부작용입니다. 이는 예상치 못한 동작으로 이어질 수 있으며 시스템 상태에 대한 추론을 훨씬 더 어렵게 만듭니다. - 
벤더 종속 및 이식성 문제: 트리거 구문 및 기능은 서로 다른 데이터베이스 시스템(예: MySQL, PostgreSQL, Oracle, SQL Server) 간에 크게 다를 수 있습니다. 중요한 비즈니스 로직을 트리거에 배치하면 향후 애플리케이션을 다른 데이터베이스 벤더로 마이그레이션하는 것이 매우 어려울 수 있습니다.
 
비즈니스 로직이 진정으로 속한 곳
더 유지보수 가능하고, 확장 가능하며, 테스트 가능한 시스템을 위해 비즈니스 로직은 일반적으로 애플리케이션 계층에 상주해야 합니다. 일반적으로 전용 서비스 계층, 도메인 모델 또는 애플리케이션 모듈 내에 있습니다.
order가 shipped로 표시될 때 해당 product의 stock_quantity가 감소해야 하는 요구 사항을 고려해 보겠습니다.
트리거 접근 방식 (피하기):
-- PostgreSQL 예시, 문법은 DB에 따라 다름 CREATE OR REPLACE FUNCTION decrement_stock_on_shipment() RETURNS TRIGGER AS $$ BEGIN IF NEW.status = 'shipped' AND OLD.status != 'shipped' THEN UPDATE products SET stock_quantity = stock_quantity - NEW.quantity WHERE id = NEW.product_id; END IF; RETURN NEW; END; $$ LANGUAGE plpgsql; CREATE TRIGGER update_stock_after_order_shipment AFTER UPDATE ON orders FOR EACH ROW EXECUTE FUNCTION decrement_stock_on_shipment();
이 트리거에서는 재고를 감소시키는 로직이 데이터베이스 안에 숨겨져 있습니다. products 테이블 스키마가 변경되거나 stock_quantity에 대한 새로운 규칙(예: 최소 재고 수준 확인)이 있는 경우 트리거를 업데이트해야 합니다. 디버깅 및 테스트는 번거로울 수 있습니다.
**애플리케이션 계층 접근 방식 (권장):
여기서는 로직이 애플리케이션 서비스 내에 상주합니다. Python/Django와 유사한 의사 코드를 가정해 봅시다.
# models.py class Product: id: int name: str stock_quantity: int class Order: id: int product_id: int quantity: int status: str # 예: 'pending', 'shipped', 'cancelled' # services.py class OrderService: def ship_order(self, order_id: int): order = Order.get_by_id(order_id) if order.status == 'shipped': raise ValueError("Order already shipped.") product = Product.get_by_id(order.product_id) if product.stock_quantity < order.quantity: raise ValueError("Insufficient stock for product.") # 상품 재고 업데이트 product.stock_quantity -= order.quantity product.save() # 주문 상태 업데이트 order.status = 'shipped' order.save() # 이벤트를 게시하거나, 알림을 보내는 등 # 추가 비즈니스 로직을 여기에 쉽게 확장할 수 있습니다. # 뷰/컨트롤러 계층에서는 다음을 호출합니다. # order_service = OrderService() # order_service.ship_order(order_id)
이 애플리케이션 수준 접근 방식에서는 다음과 같습니다.
- 테스트 용이성: 
OrderService.ship_order는 데이터베이스 상호 작용과 같은 종속성을 모의하여 격리된 단위 테스트가 가능합니다. - 유지보수성: 로직은 애플리케이션의 기본 언어로 작성되어 개발 팀이 더 쉽게 읽고 수정할 수 있습니다.
 - 가시성 및 제어: 애플리케이션은 단계를 명시적으로 조정하여 비즈니스 로직의 흐름을 명확하게 합니다. 모든 오류는 애플리케이션에서 감지되고 처리됩니다.
 - 확장성: 주문 감소 로직이 복잡해지거나 별도의 마이크로 서비스로 이동해야 하는 경우 리팩토링하기 쉽습니다.
 
트리거를 여전히 고려할 수 있는 경우 (신중하게)
일반적으로 권장되지 않지만, 제약 조건으로 강제할 수 없는 매우 저수준의 스키마 독립적인 데이터 무결성 규칙을 강제하기 위해 트리거를 고려할 수 있는 몇 가지 틈새 시나리오가 있습니다. 예시는 다음과 같습니다.
- 감사 로깅: 특정 테이블의 모든 변경 사항을 자동으로 기록합니다. 그러나 이조차도 변경 데이터 캡처(CDC) 메커니즘이나 애플리케이션 수준 로깅으로 더 잘 처리될 수 있습니다.
 - 복잡한 참조 무결성: 표준 외래 키 제약 조건이 불충분한 시나리오.
 - 파생 열 (사전 계산 값): 최신 데이터베이스의 뷰 또는 계산 열로 더 잘 처리되는 경우가 많습니다.
 
이러한 경우에도 트리거를 사용하기로 한 결정은 잠재적인 장기적인 비용을 완전히 내면화하고 극도로 신중하게 내려야 합니다.
결론
데이터베이스 트리거는 데이터에 가까운 로직을 즉시 내장하는 솔루션을 제공하지만, 불투명성, 테스트 및 디버깅의 어려움, 단단히 결합된 특성으로 인해 종종 심각한 유지보수 문제와 시스템 유연성 저하를 초래합니다. 비즈니스 로직은 달리 강제할 수 없는 매우 구체적이고 저수준의 데이터 무결성 규칙이 아닌 한, 애플리케이션 계층에 확실히 속합니다. 트리거에서 비즈니스 로직을 제외함으로써 개발자는 개발, 테스트 및 수명 주기 전반에 걸쳐 유지보수가 더 쉬운 더 강력하고 확장 가능하며 이해하기 쉬운 시스템을 구축할 수 있습니다. 데이터베이스 트리거의 숨겨진 복잡성보다 애플리케이션 코드의 명시적인 제어와 명확성을 선호하십시오.