https://www.youtube.com/watch?v=WXsD0ZgxjRw&t=9528s 영상 리뷰를 토대로 정리하였습니다
: 기능을 우리가 컨트롤할 수 있게 해주는, interact with software (don’t need to understand about specific implementation)
ex)
다른 소프트웨어 시스템과 통신하기 위해 따라야하는 규칙을 정의 (=메뉴판)
지정된 형식으로 요청, 명령을 받을 수 있는 수단
https://www.youtube.com/watch?v=ckSdPNKM2pY
우리가 직접 특정 컨텐츠에 대한 내용을 요구하는 코드를 짤 필요 없이 웹페이지 아이템들을 클릭함으로써 api를 요청할 수 있다.
네이버 클라우드 서비스들도 api 요청을 통해 활용할 수 있다
: URI와 HTTP method를 활용해서 자원과 행위를 표현
(URL로 어떤 자원에 접근할 것인지, method로 어떤 행위를 할 것인지 표현하여 API 설계)
: REST의 원칙을 잘 지키고 파악하기 쉽게 만들었다 = restful
: 데이터는 JSON 혹은 XML을 통해 주고 받는
활용도 good 다른 사람들도 보고 파악이 쉽게 끔
ex) --- GET /movies/inception/actors (O) --- /getTopRatedMovies (X) 이렇게 쓰지 않는다 |
*URL 내에서는 동사를 사용하지 않고 명사 형태로만 (ex. /movies, /movies/id=25)
*resource = reference of object
웹 내 하나하나 데이터(HTML, 이미지, 음성데이터 ,,) = Resource
그 resource를 uri로 표현 = element
element의 집합 = collection
*CRUD - HTTP verbs 로 resource handle
REST API를 만들기 위한 가이드라인
Use JSON as the Format for Sending and Receiving Data | XML은 더이상 많이 사용X |
Use Nouns Instead of Verbs in Endpoints | HTTP verbs |
Name Collections with Plural Nouns | instead of https://mysite.com/post/123 , use https://mysite.com/posts/123 |
Use Status Codes in Error Handling | |
Use Nesting on Endpoints to Show Relationships | 종점 URL 중첩해서 명확하게 쓰기 ex)https://mysite.com/posts/postId/comments |
Use Filtering, Sorting, and Pagination to Retrieve the Data Requested | 방대한 DB에서 값 잘 찾을 수 있도록 |
Use SSL for Security | https:// |
Be Clear with Versioning | ~/v1 ~/v2 |
Provide Accurate API Documentation |
https://www.freecodecamp.org/news/rest-api-best-practices-rest-endpoint-design-examples/
💡
curl(client url) 명령어는 프로토콜들을 이용해 URL 로 데이터를 전송하여 서버에 데이터를 보내거나 가져올때 사용하기 위한 명령줄 도구 및 라이브러리이다.
쉽게말해 예를들어 자바스크립트 환경에서 REST API(http)를 테스트하고싶다면 보통 ajax, fetch 를 이용해 요청을 보내는 것과 같이, SHELL(커맨드라인 환경)에서 REST API(http) 테스트 하고 싶으면 curl 명령어를 이용하면 된다 라고 이해하면 된다.
https://inpa.tistory.com/entry/LINUX-📚-CURL-명령어-사용법-다양한-예제로-정리
curl -X POST "https://api.twilio.com/2010-04-01/Accounts/$TWILIO_ACCOUNT_SID/Messages.json" \
--data-urlencode "From=+15017122661" \
--data-urlencode "Body=Hi there" \
--data-urlencode "To=+15558675310" \
-u $TWILIO_ACCOUNT_SID:$TWILIO_AUTH_TOKEN
twilio를 통해 보낸 문자 내용의 json 형태
{
"account_sid": "ACXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX",
"api_version": "2010-04-01",
"body": "Hi there",
"date_created": "Thu, 30 Jul 2015 20:12:31 +0000",
"date_sent": "Thu, 30 Jul 2015 20:12:33 +0000",
"date_updated": "Thu, 30 Jul 2015 20:12:33 +0000",
"direction": "outbound-api",
"error_code": null,
"error_message": null,
"from": "+15017122661",
"messaging_service_sid": "MGXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX",
"num_media": "0",
"num_segments": "1",
"price": null,
"price_unit": null,
"sid": "SMXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX",
"status": "sent",
"subresource_uris": {
"media": "/2010-04-01/Accounts/ACXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX/Messages/SMXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX/Media.json"
},
"to": "+15558675310",
"uri": "/2010-04-01/Accounts/ACXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX/Messages/SMXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX.json"
}
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<link
rel="stylesheet"
href="https://fonts.googleapis.com/css?family=Open+Sans"
/>
<style>
body {
font-family: "Open Sans", sans-serif;
margin: 5%;
}
</style>
<script src="https://cdn.jsdelivr.net/npm/vue@2.7.13"></script>
<script src="https://unpkg.com/vue-silentbox@2.3.1/dist/vue-silentbox.min.js"></script>
<title>Pick.le: Pick your pics 🥒</title>
</head>
<body>
<div id="app">
<h1>Pick.le: Pick your pics 🥒</h1>
<h2>{{ callToAction }}</h2>
<silent-box :gallery="gallery"></silent-box>
</div>
<script>
Vue.use(VueSilentbox.default);
const app = new Vue({
el: "#app",
data() {
return {
callToAction: "Submit your photos of applepie!",
gallery: [],
};
},
methods: {
async loadImages() {
// TODO: Use the Messaging API to use submitted photos
// const response = await fetch("/api/")
// TODO: Create a web-based API that matches this expected response
this.gallery = [
{
src: "",
description: "Look at this kitteh",
alt: "A kitteh",
thumbnailWidth: "200px",
},
{
src: "",
description: "Another Kitteh",
alt: "Cutie",
thumbnailWidth: "200px",
},
];
},
},
mounted() {
this.loadImages();
},
});
</script>
</body>
</html>
/incoming-message
// This is your new function. To start, set the name and path on the left.
exports.handler = function(context, event, callback) {
// Here's an example of setting up some TWiML to respond to with this function
console.log(`Incoming message: ${event.Body}`);
let twiml = new Twilio.twiml.MessagingResponse();
twiml.message('Thanks for your submission! :)');
console.log(`TwiML was ${twiml}`);
// This callback is what is returned in response to this function being invoked.
// It's really important! E.g. you might respond with TWiML here for a voice or SMS response.
// Or you might return JSON data to a studio flow. Don't forget it!
return callback(null, twiml);
};
twilio로 만든 전화번호로 문자를 받으면 어떻게 답장할지를 incoming message 함수를 통해 설정
serverless
twilio function을 통해 html 페이지를 실제 인터넷 상에 올림
web hooks ~ reverse API (api calls you)
/api/pics
exports.handler = async function (context, event, callback) {
const client = context.getTwilioClient();
const gallery = [];
/* This is the format we need to match. Here for reference
[
{
src: "https://placekitten.com/200/300",
description: "Look at this kitteh",
alt: "A kitteh",
thumbnailWidth: "200px",
},
];
*/
const messages = await client.messages.list({ to: context.TWILIO_NUMBER });
for (const message of messages) {
// You can have multiple medias on each message
const pics = await message.media().list();
for (const pic of pics) {
// Add to the gallery array, use the outer loop's message value to put the same body
// for each pic
gallery.push({
src: "https://api.twilio.com" + pic.uri.replace(".json", ""),
description: message.body,
alt: message.body,
thumbnailWidth: "200px",
});
}
}
// Twilio Function will automatically turn gallery into proper JSON and set the
// header to `application\json`
return callback(null, gallery);
};
index.html 수정
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<link
rel="stylesheet"
href="https://fonts.googleapis.com/css?family=Open+Sans"
/>
<style>
body {
font-family: "Open Sans", sans-serif;
margin: 5%;
}
</style>
<script src="https://cdn.jsdelivr.net/npm/vue@2.7.13"></script>
<script src="https://unpkg.com/vue-silentbox@2.3.1/dist/vue-silentbox.min.js"></script>
<title>Pick.le: Pick your pics 🥒</title>
</head>
<body>
<div id="app">
<h1>Pick.le: Pick your pics 🥒</h1>
<h2>{{ callToAction }}</h2>
<silent-box :gallery="gallery"></silent-box>
</div>
<script>
Vue.use(VueSilentbox.default);
const app = new Vue({
el: "#app",
data() {
return {
callToAction: "Submit your photos of burritos!",
gallery: [],
};
},
methods: {
async loadImages() {
// Our API is on the same server so we don't need to specify the host
// NOTE: fetch is asynchronous and returns a `Promise`, so we'll `await`
const response = await fetch("/api/pics");
// NOTE: The `json` method on the response object is also asynchronous
// Setting the gallery object automatically causes the API to refresh because
// we bound it to the plugin as part of the `data` API of Vue.
this.gallery = await response.json();
},
},
mounted() {
this.loadImages();
},
});
</script>
</body>
</html>
해외로 MMS전송이 안되지 않아 제대로 구현은 어려웠다
💡
비동기 처리 async
https://www.daleseo.com/js-window-fetch/
[3.5차시/16기 박민규] FastAPI-(2) (0) | 2023.04.02 |
---|---|
[3차시/15기 공도웅] Fast API (0) | 2023.03.26 |
[2차시/15기 이병주] REST API 와 GraphQL (0) | 2023.03.23 |
[1.5차시/15기 이병주] TCP/IP (0) | 2023.03.23 |
[2차시/16기 박민규] HTTP&HTTPS (0) | 2023.03.22 |
댓글 영역