오늘은 React16
에서 도입된 ErrorBoundary
에 대해 정리해보려고 한다. 대부분 내용은 리액트 공식문서를 참고하여 작성하였고, react-dev에서도 도움을 받았다.
이는 에러 처리를 더 쉽고 효과적으로 할 수 있게 도와준다. ErrorBoundary
는 하위 컴포넌트 트리에서 발생한 에러를 잡아 선언적으로 처리할 수 있는 컴포넌트를 말한다. (즉, 하위 컴포넌트 어디에서든 오류를 기록하며, 깨진 컴포넌트 대신 Fallback UI를 보여준다.)
++ fallback UI
는 보조적인 사용자 인터페이스로 의미하면 된다. 특정 이유로 주요 사용자 인터페이스가 실패하거나 보이지 않는 경우 대체 기능을 가진 UI로 사용된다.
ErrorBoundary
는 크게 두 가지 메서드를 정의하고 있다. 이 두가지 중 하나만 정의 하여도 컴포넌트 자체에 에러 경계가 된다. static getDerivedStateFromError()
는 오류에 대한 응답으로 상태를 업데이트하고 사용자에게 오류 메세지를 표시할 수 있는 기능과 함께 사용된다. (다음 렌더링에서 fallback UI
가 보이도록 한다.)
componentDidCatch()
는 서비스에 대한 오류를 기록한다.
이러한 ErrorBoundary
는 자바스크립트에서 catch {}
문과 유사하게 동작하지만 컴포넌트에만 적용된다는 차이가 있다. (기존에 처리해왔던 try-catch
문은 try로 감싼 곳의 에러를 핸들링했다면 ErrorBoundary
는 선언적으로 핸들링할 수 있다.)
ErrorBoundary
는 아래에 경우에는 에러를 핸들링하지 않는다.
오늘은 두 가지 방법으로 ErrorBoundary
를 다루어 보려고 한다.
현재 나는 VITE + REACT + TYPESCRIPT에서 프로젝트를 진행하고 있다. 먼저, 나는 src/compoents/ErrorBoundary
안에 파일을 생성해두었다.
import React from "react";
/**
* ErrorBoundary 컴포넌트의 프로퍼티를 정의합니다.
* @interface ErrorBoundaryProps
*/
interface ErrorBoundaryProps {
fallback: JSX.Element;
children: JSX.Element;
}
/**
* ErrorBoundary의 state 프로퍼티를 정의합니다.
* @interface ErrorBoundaryState
*/
interface ErrorBoundaryState {
hasError: boolean;
}
/**
* 에러를 처리하고 대체 UI를 표시하는 React 컴포넌트입니다.
* @class ErrorBoundary
* @extends {React.Component<ErrorBoundaryProps, ErrorBoundaryState>}
*/
class ErrorBoundary extends React.Component<
ErrorBoundaryProps,
ErrorBoundaryState
> {
constructor(props: ErrorBoundaryProps) {
super(props);
this.state = { hasError: false };
}
static getDerivedStateFromError() {
return { hasError: true };
}
componentDidCatch(error: Error, errorInfo: React.ErrorInfo): void {
console.error("Error caught by componentDidCatch:", error, errorInfo);
}
render() {
if (this.state.hasError) {
return this.props.fallback;
}
return this.props.children;
}
}
export default ErrorBoundary;
이렇게 만들어진 ErrorBoundary
컴포넌트를 적용해보자. 나는 임시적으로 App
컴포넌트를 감싸게 설정하였다.
ReactDOM.createRoot(document.getElementById("root")!).render(
<ErrorBoundary fallback={<h1>잘못된 접근입니다:(</h1>}>
<App />
</ErrorBoundary>
);
그리고 App
컴포넌트에서 clientKey
의 값의 존재 여부에 따라 Error를 핸들링하게 구현해보자. 아래와 같이 코드를 실행해보면 예상했던 화면을 보여줄 것이다.
const App = () => {
const clientKey = "sfaa-2agb-324d-afdv";
if (!clientKey) throw Error("clientKey is undefined");
return (
<div>
<h1>Error Boundary Test</h1>
<label> clientKey : {clientKey} </label>
</div>
);
};
export default App;
반대로 아래와 같이 코드를 작성해보자.
const App = () => {
let clientKey = undefined;
if (!clientKey) throw Error("clientKey is undefined");
return (
<div>
<h1>Error Boundary Test</h1>
<label> clientKey : {clientKey} </label>
</div>
);
};
export default App;
이렇게 위와 같이 상황에 따라서 ApiErrorBoundary
나 전역에서 관리되어야 하는 케이스를 담을 GlobalErrorBoundary
등을 생성하여 에러를 처리할 수 있다.
위에서 다룬 ErrorBoundary
는 클래스형으로 구현이 되어있다. 함수형으로 만들기에는 두 메서드가 존재하지 않기 때문에 공식문서에서도 클래스형만 다루고 있다. 구글링을 해보면서 함수형 컴포넌트에서도 이를 처리할 수 있다는 사실을 발견했다. (링크-함수형으로 ErrorBoundary)
하지만 오늘은 더 간단하게 구현 가능한 react-error-boundary
를 이용해보려고 한다.
react-error-boundary
는 ErrorBoundary
를 확장하여 개선된 기능을 제공하는 라이브러리로, 전역 에러 핸들링 및 커스텀 에러 UI를 간편하게 구현할 수 있게 해준다. 어느 위치에서든 에러를 감지하고 사용자에게 보여줄 allback UI
를 쉽게 지정하여 처리할 수 있다.
지금부터 하나씩 진행해보도록 하자. 먼저, 설치를 진행한다.(나는 npm으로 패키지 매니저를 선택했다.)
npm install --save-dev react-error-boundary
설치된 라이브러리를 통해서 ErrorBoundary
를 만들어보자. 기본적으로 아래와 같이 사용할 수 있다.
<ErrorBoundary fallbackComponent={ErrorFallback}>
<MyComponent/>
</ErrorBoundary>
예시를 들어 위와 같이 App
컴포넌트를 수정해보겠다.
import { ErrorBoundary, FallbackProps } from "react-error-boundary";
const App = () => {
return (
<div>
<h1>react-error-boundary TEST</h1>
<ErrorBoundary FallbackComponent={FallbackComponent}>
<ThrowErrorComponent name="송우든" />
</ErrorBoundary>
</div>
);
};
/**
* ErrorBoundary에 잡혔을 경우, 보여줄 Fallback 컴포넌트
* @param {FallbackProps} props error, resetErrorBoundary
* @returns {JSX.Element} Fallback UI
*/
const FallbackComponent = ({ error, resetErrorBoundary }: FallbackProps) => {
return (
<div>
<h2>잘못된 접근 방식</h2>
<p>{error.message}</p>
<button onClick={resetErrorBoundary}>다시 시도</button>
</div>
);
};
/**
* Error가 발생할 수 있는 컴포넌트
* @param {{[key:string] : string}} props name
* @returns {JSX.Element} Error가 없을 때 보여질 UI
*/
const ThrowErrorComponent = ({ name }: { [key: string]: string }) => {
if (name === "") throw Error("[ThrowErrorComponent] : Invalid name");
return <h1>Hello, {name}!</h1>;
};
export default App;
name
에 값이 빈 문자열이라면 FallbackComponent
컴포넌트가 다음과 같이 나타난다.
반대로 name
이 빈 문자열이 아니라면 올바르게 동작하는 것을 확인할 수 있다.
그리고 마지막으로 react-error-boundary
에는 useErrorBoundary
라는 훅으로 제공하는데 이 훅을 이용해 함수형 컴포넌트 내부에서 에러를 핸들링할 수 있다. 아래와 같이 비동기 처리를 진행해야할 때, 에러 핸들링이 필수적으로 필요하다.
import React from 'react';
import { useErrorBoundary } from "react-error-boundary";
const MyComponent = () => {
const { showBoundary } = useErrorBoundary();
const fetchData = async () => {
try {
const result = await fetch("https://example.com/data");
const data = await result.json();
console.log(data);
} catch (error) {
showBoundary(error);
}
};
return (
<div>
<button onClick={fetchData}>데이터 가져오기</button>
</div>
);
};
export default MyComponent;
useErrorBoundary
훅을 통해서 handleError함수를 얻어 catch
블록 내에서 발생한 에러를 ErrorBoundary
에 전달하여 Fallback UI
를 활성화시킬 수 있다.
요즘 이것저것 다양한 분야를 공부해 보려고 한다. 그 동안 몰랐던 프론트엔드 관린 지식이나 기술은 물론 IT라는 분야의 관심을 가지고 학습을 진행하고 있다. 사실 이번 소프트웨어 박람회를 다녀오며 많은 생각을 가지게 되었다. 특히, 보다 나은 사용자 경험과 안전한 서비스를 만들기 위해 개발자로서 어떤 노력을 할 수 있는지, 그 동안 어떤식으로 처리해 왔는지를 고민하게 됐다. 이번 기회에 에러 핸들링에 대해 정확하게 학습하고 적용해볼 수 있게 노력해야겠다.