HTTP
의 장점을 최대한 활용할 수 있는 아키텍처로서 REST가 소개되었고, 이는 HTTP 프로토콜을 의도에 맞게 디자인하도록 유도하고 있다.
REST의 기본 원칙을 성실히 지킨 서비스 디자인을 RESTful
이라고 표현한다.
- 즉, REST는 HTTP를 기반으로 클라이언트가 서버의 리소스에 접근하는 방식을 규정한 아키텍처고,
REST API
는 REST를 기반으로 서비스 API를 구현한 것을 의미한다.
REST API는 자원
, 행위
, 표현
의 3가지 요소로 구성된다.
REST는 자체 표현 구조로 구성되어 REST API만
으로 HTTP 요청의 내용을 이해할 수 있다.
자원
: 자원 (URI)행위
: 자원에 대한 행위 (HTTP 요청 메서드)표현
: 자원에 대한 행위의 구체적 내용 (페이로드)REST에서 가장 중요한 기본적인 원칙
은 두 가지다.
URI
는 리소스를 표현해야 한다. HTTP 요청 메서드
로 표현한다.get
같은 행위에 대한 표현이 들어가서는 안 된다.# 나쁜 예시
GET /getTodos/1
GET /todos/show/1
# 좋은 예시
GET todos/1
HTTP
요청 메서드는 클라이언트가 서버에게 요청의 종류와 목적을 알리는 방법이다.CRUD
를 구현한다.# 나쁜 예시
GET /todos/delete/1
# 좋은 예시
DELETE todos/1
JSON Server를 사용해 가상 REST API 서버를 구축하여 HTTP 요청을 전송하고 응답을 받을 수 있다.
JSON Server
는 json 파일을 사용하여 가상 API 서버를 구축할 수 있는 툴
이다.설치 방법
npm install json-server --save-dev
db.json
은 리소스를 제공하는 데이터베이스 역할을 한다.{
"todos": [
{
"id": 1,
"content": "HTML",
"completed": true
},
{
"id": 2,
"content": "CSS",
"completed": true
},
{
"id": 3,
"content": "JavaScript",
"completed": false
}
]
}
JSON Server
를 실행한다.JSON Server
가 데이터베이스 역할을 하는 db.json
파일의 변경을 감지하게 하려면 watch
옵션을 추가한다.json-server --watch db.json
package.json
파일의 scripts
를 다음과 같이 수정할 수 있다.{
"name": "json-server-exam",
"version": "1.0.0",
"scripts": {
"start": "json-server --watch db.json"
},
"devDependencies": {
"json-server": "^0.16.1"
}
}
npm start
명령어를 입력하여 JSON Server
를 실행한다.todos 리소스에서 모든 todo를 취득한다.
public 폴더에 get_index.html
을 추가하고 브라우저에서
http://localhost:3000/get_index.html
로 접속한다.
<!DOCTYPE html>
<html>
<body>
<pre></pre>
<script>
const xhr = new XMLHttpRequest();
// HTTP 요청 초기화
// 모든 todo 취득
xhr.open('GET', '/todos');
// 특정 todo 취득
xhr.open('GET', '/todos/1');
// HTTP 요청 전송
xhr.send();
// load 이벤트는 요청이 성공적으로 완료된 경우 발생한다.
xhr.onload = () => {
// 정상적으로 응답이 오면 응답을 객체로 변환한다.
if(xhr.response === 200){
document.querySelector('pre').textContent = xhr.response;
}else{
console.log('Error', xhr.status, xhr.statusText);
}
};
</script>
</body>
</html>
todos
리소스에 새로운 todo
를 생성한다.
post
요청 시에는 setRequestHeader
메서드를 사용하여 요청 몸체에 담아 서버로 전송할 페이로드의 MIME 타입을 지정해야 한다.
public 폴더에 다음 post.html을 추가하고 브라우저에서
http://localhost:3000/post_index.html
로 접속한다.
<!DOCTYPE html>
<html>
<body>
<pre></pre>
<script>
const xhr = new XMLHttpRequest();
// HTTP 요청 초기화
xhr.open('POST', '/todos');
// 요청 몸체에 담아 서버로 전송한 페이로드의 MIME 타입을 지정
xhr.setRequestHeader('content-type', 'application/json');
// HTTP 요청 전송
xhr.send(JSON.stringify({id: 4, content: 'Angular', completed: false}));
// load 이벤트는 요청이 성공적으로 완료된 경우 발생한다.
xhr.onload = () => {
// 정상적으로 응답이 오면 응답을 객체로 변환한다.
if(xhr.status === 200){
document.querySelector('pre').textContent = xhr.response;
}else{
console.log('Error', xhr.status, xhr.statusText);
}
};
</script>
</body>
</html>
PUT
요청 시에는 setRequestHeader
메서드를 사용하여 요청 몸체에 담아 서버로 전송할 페이로드의 MIME 타입을 지정해야 한다.
public
폴더에 다음 put.html을 추가하고 브라우저에서
http://localhost:3000/put_index.html
로 접속한다.
<!DOCTYPE html>
<html>
<body>
<pre></pre>
<script>
const xhr = new XMLHttpRequest();
// HTTP 요청 초기화
xhr.open('PUT', '/todos/4');
// 요청 몸체에 담아 서버로 전송한 페이로드의 MIME 타입을 지정
xhr.setRequestHeader('content-type', 'application/json');
// HTTP 요청 전송
xhr.send(JSON.stringify({id: 4, content: 'React', completed: false}));
// load 이벤트는 요청이 성공적으로 완료된 경우 발생한다.
xhr.onload = () => {
// 정상적으로 응답이 오면 응답을 객체로 변환한다.
if(xhr.status === 200){
document.querySelector('pre').textContent = xhr.response;
}else{
console.log('Error', xhr.status, xhr.statusText);
}
};
</script>
</body>
</html>
PATCT
요청 시에는 setRequestHeader
메서드를 사용하여 요청 몸체에 담아 서버로 전송할 페이로드의 MIME 타입을 지정해야 한다.
public
폴더에 다음 patch.html
을 추가하고 브라우저에서
http://localhost:3000/patch_index.html
로 접속한다.
<!DOCTYPE html>
<html>
<body>
<pre></pre>
<script>
const xhr = new XMLHttpRequest();
// HTTP 요청 초기화
xhr.open('PATCH', '/todos/4');
// 요청 몸체에 담아 서버로 전송한 페이로드의 MIME 타입을 지정
xhr.setRequestHeader('content-type', 'application/json');
// HTTP 요청 전송
xhr.send(JSON.stringify({completed: true}));
// load 이벤트는 요청이 성공적으로 완료된 경우 발생한다.
xhr.onload = () => {
// 정상적으로 응답이 오면 응답을 객체로 변환한다.
if(xhr.status === 200){
document.querySelector('pre').textContent = xhr.response;
}else{
console.log('Error', xhr.status, xhr.statusText);
}
};
</script>
</body>
</html>
delete.html
을 추가하고 브라우저에서 http://localhost:3000/delete_index.html
로 접속한다.<!DOCTYPE html>
<html>
<body>
<pre></pre>
<script>
const xhr = new XMLHttpRequest();
// HTTP 요청 초기화
xhr.open('DELETE', '/todos/4');
// 요청 몸체에 담아 서버로 전송한 페이로드의 MIME 타입을 지정
xhr.setRequestHeader('content-type', 'application/json');
// HTTP 요청 전송
xhr.send();
// load 이벤트는 요청이 성공적으로 완료된 경우 발생한다.
xhr.onload = () => {
// 정상적으로 응답이 오면 응답을 객체로 변환한다.
if(xhr.status === 200){
document.querySelector('pre').textContent = xhr.response;
}else{
console.log('Error', xhr.status, xhr.statusText);
}
};
</script>
</body>
</html>