RESTλ HTTPλ₯Ό κΈ°λ°μΌλ‘ ν΄λΌμ΄μΈνΈκ° μλ²μ 리μμ€μ μ κ·Όνλ λ°©μμ κ·μ ν μν€ν μ²κ³ , REST APIλ RESTλ₯Ό κΈ°λ°μΌλ‘ μλΉμ€ APIλ₯Ό ꡬνν κ²μ μλ―Ένλ€.
REST APIλ μμ(resource), νμ(verb), νν(representations)μ 3κ°μ§ μμλ‘ κ΅¬μ±λλ€. RESTλ μ체 νν ꡬ쑰(self-de-scriptiveness)λ‘ κ΅¬μ±λμ΄ REST APIλ§μΌλ‘ HTTP μμ²μ λ΄μ©μ μ΄ν΄ν μ μλ€.
κ΅¬μ± μμ | λ΄μ© | νν λ°©λ² |
---|---|---|
μμ(resource) | μμ | URI(μλν¬μΈνΈ) |
νμ(verb) | μμμ λν νμ | HTTP μμ² λ©μλ |
νν(representations) | μμμ λν νμμ ꡬ체μ λ΄μ© | νμ΄λ‘λ |
URIλ 리μμ€λ₯Ό ννν΄μΌ νλ€.
# bad
GET /getTodos/1
GET /todos/show/1
# good
GET /todos/1
리μμ€μ λν νμλ HTTP μμ² λ©μλλ‘ νννλ€.
HTTP μμ² λ©μλλ ν΄λΌμ΄μΈνΈκ° μλ²μκ² μμ²μ μ’ λ₯μ λͺ©μ (리μμ€μ λν νμ)μ μ리λ λ°©λ²μΌλ‘ μ£Όλ‘ 5κ°μ μμ² λ©μλλ₯Ό μ¬μ©ν΄ CRUDλ₯Ό ꡬννλ€.
HTTP μμ² λ©μλ | μ’ λ₯ | λͺ©μ | νμ΄λ‘λ |
---|---|---|---|
GET | index/retrieve | λͺ¨λ /νΉμ 리μμ€ μ·¨λ | X |
POST | create | 리μμ€ μμ± | O |
PUT | replace | 리μμ€μ μ 체 κ΅μ²΄ | O |
PATCH | modify | 리μμ€μ μΌλΆ μμ | O |
DELETE | delete | λͺ¨λ /νΉμ 리μμ€ μμ | X |
# bad
GET /todos/delete/1
# good
DELETE /todos/1
$npm install json-server --sava-D
## db.json
{
"todos": [
{
"id": 1,
"content": "HTML",
"completed": true
},
{
"id": 2,
"content": "CSS",
"completed": false
},
{
"id": 3,
"content": "Javascript",
"completed": true
}
]
}
## κΈ°λ³Έν¬νΈ(3000) μ¬μ© / watch μ΅μ
μ μ©
$json-server --watch db.json
## ν¬νΈ λ³κ²½ / watch μ΅μ
μ μ©
$json-server --watch db.json --port 5000
{
"name": "json-server-exam",
"version": "1.0.0",
"scripts": {
"start": "json-server --watch db.json"
},
"devDependencies": {
"json-server": "^0.16.1"
}
}
todos 리μμ€μμ λͺ¨λ todoλ₯Ό μ·¨λ(index)νλ€.
<!DOCTYPE html>
<html>
<body>
<pre></pre>
<script>
// XMLHttpRequest κ°μ²΄ μμ±
const xhr = new XMLHttpRequest();
// HTTP μμ² μ΄κΈ°ν
// todos 리μμ€μμ λͺ¨λ todoλ₯Ό μ·¨λ(index)
xhr.open('GET', '/todos');
// HTTP μμ² μ μ‘
xhr.send();
// load μ΄λ²€νΈλ μμ²μ΄ μ±κ³΅μ μΌλ‘ μλ£λ κ²½μ° λ°μνλ€.
xhr.onload = () => {
// status νλ‘νΌν° κ°μ΄ 200μ΄λ©΄ μ μμ μΌλ‘ μλ΅λ μνλ€.
if (xhr.status === 200) {
document.querySelector('pre').textContent = xhr.response;
} else {
console.error('Error', xhr.status, xhr.statusText);
}
};
</script>
</body>
</html>
todos 리μμ€μ μλ‘μ΄ todoλ₯Ό μμ±νλ€. POST μμ²μμλ setRequestHeader λ©μλλ₯Ό μ¬μ©νμ¬ μμ² λͺΈμ²΄μ λ΄μ μλ²λ‘ μ μ‘ν νμ΄λ‘λμ MINE νμ μ μ§μ νλ€.
<!DOCTYPE html>
<html>
<body>
<pre></pre>
<script>
// XMLHttpRequest κ°μ²΄ μμ±
const xhr = new XMLHttpRequest();
// HTTP μμ² μ΄κΈ°ν
// todos 리μμ€μ μλ‘μ΄ todoλ₯Ό μμ±
xhr.open('POST', '/todos');
// μμ² λͺΈμ²΄μ λ΄μ μλ²λ‘ μ μ‘ν νμ΄λ‘λμ MIME νμ
μ μ§μ
xhr.setRequestHeader('content-type', 'application/json');
// HTTP μμ² μ μ‘
// μλ‘μ΄ todoλ₯Ό μμ±νκΈ° μν΄ νμ΄λ‘λλ₯Ό μλ²μ μ μ‘ν΄μΌ νλ€.
xhr.send(JSON.stringify({ id: 4, content: 'Angular', completed: false }));
// load μ΄λ²€νΈλ μμ²μ΄ μ±κ³΅μ μΌλ‘ μλ£λ κ²½μ° λ°μνλ€.
xhr.onload = () => {
// status νλ‘νΌν° κ°μ΄ 200(OK) λλ 201(Created)μ΄λ©΄ μ μμ μΌλ‘ μλ΅λ μνλ€.
if (xhr.status === 200 || xhr.status === 201) {
document.querySelector('pre').textContent = xhr.response;
} else {
console.error('Error', xhr.status, xhr.statusText);
}
};
</script>
</body>
</html>
νΉμ 리μμ€ μ 체λ₯Ό κ΅μ²΄ν λ μ¬μ©νλ€.
<!DOCTYPE html>
<html>
<body>
<pre></pre>
<script>
// XMLHttpRequest κ°μ²΄ μμ±
const xhr = new XMLHttpRequest();
// HTTP μμ² μ΄κΈ°ν
// todos 리μμ€μμ idλ‘ todoλ₯Ό νΉμ νμ¬ idλ₯Ό μ μΈν 리μμ€ μ 체λ₯Ό κ΅μ²΄
xhr.open('PUT', '/todos/4');
// μμ² λͺΈμ²΄μ λ΄μ μλ²λ‘ μ μ‘ν νμ΄λ‘λμ MIME νμ
μ μ§μ
xhr.setRequestHeader('content-type', 'application/json');
// HTTP μμ² μ μ‘
// 리μμ€ μ 체λ₯Ό κ΅μ²΄νκΈ° μν΄ νμ΄λ‘λλ₯Ό μλ²μ μ μ‘ν΄μΌ νλ€.
xhr.send(JSON.stringify({ id: 4, content: 'React', completed: true }));
// load μ΄λ²€νΈλ μμ²μ΄ μ±κ³΅μ μΌλ‘ μλ£λ κ²½μ° λ°μνλ€.
xhr.onload = () => {
// status νλ‘νΌν° κ°μ΄ 200μ΄λ©΄ μ μμ μΌλ‘ μλ΅λ μνλ€.
if (xhr.status === 200) {
document.querySelector('pre').textContent = xhr.response;
} else {
console.error('Error', xhr.status, xhr.statusText);
}
};
</script>
</body>
</html>
PATCHλ νΉμ 리μμ€μ μΌλΆλ₯Ό μμ ν λ μ¬μ©νλ€.
<!DOCTYPE html>
<html>
<body>
<pre></pre>
<script>
// XMLHttpRequest κ°μ²΄ μμ±
const xhr = new XMLHttpRequest();
// HTTP μμ² μ΄κΈ°ν
// todos 리μμ€μ idλ‘ todoλ₯Ό νΉμ νμ¬ completedλ§ μμ
xhr.open('PATCH', '/todos/4');
// μμ² λͺΈμ²΄μ λ΄μ μλ²λ‘ μ μ‘ν νμ΄λ‘λμ MIME νμ
μ μ§μ
xhr.setRequestHeader('content-type', 'application/json');
// HTTP μμ² μ μ‘
// 리μμ€λ₯Ό μμ νκΈ° μν΄ νμ΄λ‘λλ₯Ό μλ²μ μ μ‘ν΄μΌ νλ€.
xhr.send(JSON.stringify({ completed: false }));
// load μ΄λ²€νΈλ μμ²μ΄ μ±κ³΅μ μΌλ‘ μλ£λ κ²½μ° λ°μνλ€.
xhr.onload = () => {
// status νλ‘νΌν° κ°μ΄ 200μ΄λ©΄ μ μμ μΌλ‘ μλ΅λ μνλ€.
if (xhr.status === 200) {
document.querySelector('pre').textContent = xhr.response;
} else {
console.error('Error', xhr.status, xhr.statusText);
}
};
</script>
</body>
</html>
todos 리μμ€μμ idλ₯Ό μ¬μ©νμ¬ todoλ₯Ό μμ νλ€.
<!DOCTYPE html>
<html>
<body>
<pre></pre>
<script>
// XMLHttpRequest κ°μ²΄ μμ±
const xhr = new XMLHttpRequest();
// HTTP μμ² μ΄κΈ°ν
// todos 리μμ€μμ idλ₯Ό μ¬μ©νμ¬ todoλ₯Ό μμ νλ€.
xhr.open('DELETE', '/todos/4');
// HTTP μμ² μ μ‘
xhr.send();
// load μ΄λ²€νΈλ μμ²μ΄ μ±κ³΅μ μΌλ‘ μλ£λ κ²½μ° λ°μνλ€.
xhr.onload = () => {
// status νλ‘νΌν° κ°μ΄ 200μ΄λ©΄ μ μμ μΌλ‘ μλ΅λ μνλ€.
if (xhr.status === 200) {
document.querySelector('pre').textContent = xhr.response;
} else {
console.error('Error', xhr.status, xhr.statusText);
}
};
</script>
</body>
</html>