0. axios란?
axios는 node.js와 브라우저를 위한 Promise 기반의 HTTP 클라이언트다. 즉 브라우저 뿐만 아니라 NodeJS 위에서도 동작한다는 것이며, 브라우저에서는 XMLHttpRequests를 사용하고 NodeJS에서는 http 모듈을 사용한다.
axios는 Promise를 기반으로 JSON으로 데이터가 자동 변환된다는 점이 있어 기본으로 제공해주는 fetch에 비하여 편리한 점이 많아서 가장 많이 사용되는 HTTP 라이브러리 중 하나다.
axios를 이용하면 요청에 대한 인터셉트와 데이터 변환, 요청 취소 등의 기능도 활용할 수 있다.
오늘 이 포스터에서는 기본적은 axios 사용 방법과 React에서 활용할 수 있는 방법에 대하여 다룬다.
1. axios 시작하기
먼저 axios를 설치해야한다.
yarn add axios
axios를 사용하여 간단하게 GET, POST, DELETE, PUT 등의 HTTP 요청을 보낼 수 있다.
axios.get('/hello')
axios.post('/hello', {
name: 'pinomaker',
})
결과는 Promise 객체로 반환되며, fetch와 달리 JSON으로 알아서 변환되기에 파싱을 하지 않아도 된다.
axios
.get('/hello')
.then((response) => {
console.log(response)
})
.catch((error) => {
console.error(error)
})
혹은 try catch와 async/await을 이용해서도 구현이 가능하다.
const callApi = async () => {
try {
const response = await axios.get('/hello')
console.log(response)
} catch (error) {
console.error(error)
}
}
그 이외에도 config나 header 설정도 손쉽게 가능하다.
{
// `url`은 요청에 사용될 서버 URL입니다.
url: '/user',
// `method`는 요청을 생성할때 사용되는 메소드입니다.
method: 'get', // 기본값
// `url`이 절대값이 아닌 경우 `baseURL`은 URL 앞에 붙습니다.
// 상대적인 URL을 인스턴스 메서드에 전달하려면 `baseURL`을 설정하는 것은 편리합니다.
baseURL: 'https://some-domain.com/api',
// `transformRequest`는 요청 데이터를 서버로 전송하기 전에 변경할 수 있게 해줍니다.
// 이것은 'PUT', 'POST', 'PATCH', 'DELETE' 메소드에서만 적용됩니다.
// 마지막 함수는 Buffer, ArrayBuffer, FormData 또는 Stream의 인스턴스 또는 문자열을 반환해야 합니다.
// 헤더 객체를 수정할 수 있습니다.
transformRequest: [function (data, headers) {
// 데이터를 변환하려는 작업 수행
return data;
}],
// `transformResponse`는 응답 데이터가 then/catch로 전달되기 전에 변경할 수 있게 해줍니다.
transformResponse: [function (data) {
// 데이터를 변환하려는 작업 수행
return data;
}],
// `headers`는 사용자 지정 헤더입니다.
headers: {'X-Requested-With': 'XMLHttpRequest'},
// `params`은 요청과 함께 전송되는 URL 파라미터입니다.
// 반드시 일반 객체나 URLSearchParams 객체여야 합니다.
// 참고: null이나 undefined는 URL에 렌더링되지 않습니다.
params: {
ID: 12345
},
// `paramsSerializer`는 `params`의 시리얼라이즈를 담당하는 옵션 함수입니다.
// (예: https://www.npmjs.com/package/qs, http://api.jquery.com/jquery.param/)
paramsSerializer: function (params) {
return Qs.stringify(params, {arrayFormat: 'brackets'})
},
// `data`는 요청 바디로 전송될 데이터입니다.
// 'PUT', 'POST', 'PATCH', 'DELETE' 메소드에서만 적용 가능합니다.
// `transformRequest`가 설정되지 않은 경우 다음 타입 중 하나여야 합니다.
// - string, plain object, ArrayBuffer, ArrayBufferView, URLSearchParams
// - 브라우저 전용: FormData, File, Blob
// - Node 전용: Stream, Buffer
data: {
firstName: 'Fred'
},
// 바디로 전송하는 데이터의 대안 문법
// POST 메소드
// 키가 아닌 값만 전송됩니다.
data: 'Country=Brasil&City=Belo Horizonte',
// `timeout`은 요청이 시간 초과되기 전의 시간(밀리초)을 지정합니다.
// 요청이 `timeout`보다 오래 걸리면 요청이 중단됩니다.
timeout: 1000, // 기본값은 `0` (타임아웃 없음)
// `withCredentials`은 자격 증명을 사용하여 사이트 간 액세스 제어 요청을 해야 하는지 여부를 나타냅니다.
withCredentials: false, // 기본값
// `adapter`'은 커스텀 핸들링 요청을 처리할 수 있어 테스트가 쉬워집니다.
// 유효한 Promise 응답을 반환해야 합니다. (lib/adapters/README.md 참고)
adapter: function (config) {
/* ... */
},
// `auth`는 HTTP Basic 인증이 사용되며, 자격 증명을 제공합니다.
// `auth`를 사용하면, `Authorization` 헤더가 설정되어 `headers`를 사용하여 설정한 기존의 `Authorization` 사용자 지정 헤더를 덮어씁니다.
// 이 파라미터를 통해 HTTP Basic 인증만 구성할 수 있음을 참고하세요.
// Bearer 토큰 등의 경우 `Authorization` 사용자 지정 헤더를 대신 사용합니다.
auth: {
username: 'janedoe',
password: 's00pers3cret'
},
// `responseType`은 서버에서 받는 데이터의 타입입니다.
// 옵션: 'arraybuffer', 'document', 'json', 'text', 'stream'
// 브라우저 전용: 'blob'
responseType: 'json', // 기본값
// `responseEncoding`은 응답 디코딩에 사용할 인코딩입니다.
// Node.js 전용
// 참고: 클라이언트 사이드 요청 또는 `responseType`이 'stream'이면 무시합니다.
responseEncoding: 'utf8', // 기본값
// `xsrfCookieName`은 xsrf 토큰 값으로 사용할 쿠키의 이름입니다.
xsrfCookieName: 'XSRF-TOKEN', // 기본값
// `xsrfHeaderName`은 xsrf 토큰 값을 운반하는 HTTP 헤더의 이름입니다.
xsrfHeaderName: 'X-XSRF-TOKEN', // 기본값
// `onUploadProgress`는 업로드 진행 이벤트를 처리합니다.
// 브라우저 전용
onUploadProgress: function (progressEvent) {
// 업로드 진행 이벤트 작업 수행
},
// `onDownloadProgress`는 다운로드로드 진행 이벤트를 처리합니다.
// 브라우저 전용
onDownloadProgress: function (progressEvent) {
// 다운로드 진행 이벤트 작업 수행
},
// `maxContentLength`는 node.js에서 허용되는 http 응답 콘텐츠의 최대 크기를 바이트 단위로 정의합니다.
maxContentLength: 2000,
// `maxBodyLength`는 허용되는 http 요청 콘텐츠의 최대 크기를 바이트 단위로 정의합니다.
// Node.js 전용
maxBodyLength: 2000,
// `validateStatus`는 지정된 HTTP 응답 상태 코드에 대한 Promise를 이행할지 또는 거부할지 여부를 정의합니다.
// 만약 `validateStatus`가 true를 반환하면(또는 'null' 또는 'undefined'으로 설정) Promise는 이행됩니다.
// 그렇지 않으면, 그 Promise는 거부될 것이다.
validateStatus: function (status) {
return status >= 200 && status < 300; // 기본값
},
// `maxRedirects`는 node.js에서 리디렉션 최대값을 정의합니다.
// 0으로 설정하면 리디렉션되지 않습니다.
maxRedirects: 5, // 기본값
// `socketPath`는 node.js에서 사용될 UNIX 소켓을 정의합니다.
// 예: '/var/run/docker.sock' 도커 데몬에 요청을 보냅니다.
// 오직 `socketPath` 또는 `proxy`만 지정할 수 있습니다.
// 둘 다 지정되면 `socketPath`가 사용됩니다.
socketPath: null, // 기본값
// `httpAgent`와 `httpsAgent`는 각각 node.js에서 http 및 https 요청을 수행할 때 사용할 사용자 지정 에이전트를 정의합니다.
// 이렇게 하면 기본적으로 활성화되지 않은 `keepAlive`와 같은 옵션을 추가할 수 있습니다.
httpAgent: new http.Agent({ keepAlive: true }),
httpsAgent: new https.Agent({ keepAlive: true }),
// `proxy`는 프록시 서버의 호스트이름, 포트, 프로토콜을 정의합니다.
// 기존의 `http_proxy`와 `https_proxy` 환경 변수를 사용하여
// 프록시를 정의할 수도 있습니다.
// 프록시 구성에 환경 변수를 사용하는 경우, 'no_proxy' 환경 변수를
// 쉼표로 구분된 프록시가 되지 않는 도메인 목록으로 정의할 수도 있습니다.
// `false`를 사용하면 프록시를 사용하지 않고, 환경 변수를 무시합니다.
// `auth`는 프록시에 연결하는데 HTTP Basic auth를 사용해야 함을 나타내며,
// 자격 증명을 제공합니다. 그러면 `Proxy-Authorization` 헤더가 설정되고,
// `headers`를 사용하여 기존의 `Proxy-Authorization` 사용자 지정 해더를 덮어씁니다.
// 만약 프록시 서버가 HTTPS를 사용한다면, 프로토콜을 반드시 `https`로 설정해야 합니다.
proxy: {
protocol: 'https',
host: '127.0.0.1',
port: 9000,
auth: {
username: 'mikeymike',
password: 'rapunz3l'
}
},
// `cancelToken`은 요청을 취소하는 데 사용할 수 있는 취소 토큰을 지정합니다.
// (자세한 내용은 요청 취소 섹션 참조)
cancelToken: new CancelToken(function (cancel) {
}),
// `decompress`는 응답 바디의 자동 압축 해제 여부를 나타냅니다.
// `true`로 설정하면, 압축 해제된 모든 응답에서 'content-encoding' 헤더도 제거됩니다.
// Node.js 전용 (XHR은 압축 해제할 수 없습니다)
decompress: true // 기본값
}
더 자세한 내용은 공식 사이트를 참고하자.
2. axios 인스턴스 생성하기
아래와 같이 axios의 create 메서드를 사용하면 axios 인스턴스를 생성할 수 있으며 이를 이용하여 공통 속성이나 인터셉터 등을 설정할 수 있다.
const axiosInstance = axios.create({
baseURL: '/api',
})
아래와 같이 HTTP 요청할 때와 응답을 받을 때에 인터셉터를 이용하여 핸들링을 할 수 있는 것이 axios의 장점 중 하나다. 이를 이용하여 JWT와 같은 토큰을 Request 인터셉터에서 삽입하여 한 번에 처리하거나, Response 인터셉터에서 401, 403이 던져지면 Refresh Token을 이용하여 Access Token을 재발급 받는 로직을 추가할 수도 있다.
// 요청 인터셉터 추가하기
axiosInstance.interceptors.request.use(
function (config) {
// 요청이 전달되기 전에 작업 수행
return config
},
function (error) {
// 요청 오류가 있는 작업 수행
return Promise.reject(error)
},
)
// 응답 인터셉터 추가하기
axiosInstance.interceptors.response.use(
function (response) {
// 2xx 범위에 있는 상태 코드는 이 함수를 트리거 합니다.
// 응답 데이터가 있는 작업 수행
return response
},
function (error) {
// 2xx 외의 범위에 있는 상태 코드는 이 함수를 트리거 합니다.
// 응답 오류가 있는 작업 수행
return Promise.reject(error)
},
)
3. React에서 Axios 객체 제대로 이용하기
axios의 create 메서드를 이용하여 axiosInstance를 생성하고 이를 이용하여 각 GET, POST, PUT, DELETE, PATCH 5가지 함수를 생성한다. 해당 함수는 Response에 대한 타입을 받아서 TS에서 타입 추론을 이용할 수 있게 한다.
/src/api/util/index.ts
import { CommonResponse } from '@/types/common'
import axios, { AxiosRequestConfig, AxiosResponse } from 'axios'
const axiosInstance = axios.create({})
export const Get = async <T>(
url: string,
config?: AxiosRequestConfig,
): Promise<AxiosResponse<CommonResponse<T>>> => {
const response = await axiosInstance.get(url, config)
return response
}
export const Post = async <T>(
url: string,
data?: any,
config?: AxiosRequestConfig,
): Promise<AxiosResponse<CommonResponse<T>>> => {
const response = await axiosInstance.post(url, data, config)
return response
}
export const Put = async <T>(
url: string,
data?: any,
config?: AxiosRequestConfig,
): Promise<AxiosResponse<CommonResponse<T>>> => {
const response = await axiosInstance.put(url, data, config)
return response
}
export const Patch = async <T>(
url: string,
data?: any,
config?: AxiosRequestConfig,
): Promise<AxiosResponse<CommonResponse<T>>> => {
const response = await axiosInstance.patch(url, data, config)
return response
}
export const Delete = async <T>(
url: string,
data?: any,
config?: AxiosRequestConfig,
): Promise<AxiosResponse<CommonResponse<T>>> => {
const response = await axiosInstance.delete(url, config)
return response
}
그리고 Backend와의 소통할 때 사용하는 HTTP Response 객체를 interface로 선언하여 사용한다.
// ** /src/types/common.d.ts
export interface CommonResponse<T> {
data?: T
}
자 이제 axios 사용 준비를 다 했으니 로그인을 예시로 들어 작성해본다.
먼저 사용할 타입을 선언한다.
// ** /src/types/auth.d.ts
import { CommonResponse } from './common'
export interface RequestLoginParams {
username: string
password: string
}
export interface ResponseLogin extends CommonResponse {
data: {
accessToken: string
refreshToken: string
}
}
그리고 api 폴더에 도메인을 기준으로 폴더를 생성하여 서비스 파일을 만든다.
// ** src/api/auth/index.ts
import { RequestLoginParams, ResponseLogin } from '@/types/auth'
import { Post } from '../util'
// ** Type Imports
const authService = {
loginApi: async (data: RequestLoginParams) => {
const response = await Post<ResponseLogin>('/auth', data)
return response
},
}
export const { loginApi } = authService
그리고 LoginPage에서 아래와 같이 호출하면 된다.
import { loginApi } from '@/api'
import { RequestLoginParams } from '@/types/auth'
import { ChangeEvent, useState } from 'react'
const LoginPage = () => {
const [user, setUser] = useState<RequestLoginParams>({
username: '',
password: '',
})
const onChange = (e: ChangeEvent<HTMLInputElement>) => {
setUser((cur) => ({ ...cur, [e.target.name]: e.target.value }))
}
const handleLogin = () => {
loginApi(user)
.then((response) => {
console.log(response)
})
.catch((error) => {
console.error(error)
})
}
return (
<div>
<input
type="text"
name="username"
value={user.username}
onChange={onChange}
/>
<input
type="password"
name="password"
value={user.password}
onChange={onChange}
/>
<button onClick={handleLogin}>LOGIN</button>
</div>
)
}
export default LoginPage
4. 마무리하며..
최근에는 axios를 대체할 수 있는 것들이 많이 나오고 있다. React Query, React-Form, Redux Toolkit Query 등.. 필자도 사실 axios보다는 Redux Toolkit Query를 즐겨 사용하긴 한다. 이러한 데이터 fetch 라이브러리들은 캐싱, 핸들러 등 다양한 기능을 제공하여 좋고 Redux Toolkit Query는 Redux 상태와 바로 연동할 수 있는 장점이 있긴 하지만, axios도 잘 사용하면 간단한 프로젝트에서 매우 좋은 것 같다.
참고 사이트
'F.E > React' 카테고리의 다른 글
Minio에 파일 업로드하기(with aws-sdk) (0) | 2023.11.29 |
---|---|
MSW로 Mocking(1) - Brower (0) | 2023.09.23 |
[React] Cookie + JWT + axios 이용하여 JWT 인증하기. (0) | 2022.09.08 |
[React] useReducer에 대해서 알아보자. (0) | 2022.08.26 |
[React] useCallback 사용하기. (0) | 2022.08.26 |