http://devcoding.tistory.com/24 슬라이드를 풀어서 써봤는데 가독성이 그리 좋진 않네요...
Graphql과 느낌이 비슷한 netflix falcor(http://devcoding.tistory.com/28)도 있으니 모르시는 분은 한번 알아두시면 좋을듯 합니다.
Graph - 사전적으로 점과 선으로 이루어진 도형을 Graph라고 함
서로 관계가 있는 2개 또는 그 이상의 양의 상태값을 나타낸 도형
QL - SQL 비슷하다고 우선 생각하면 됨.
- 누가 만들었지?
- 언제 만들었지?
2012년부터 내부적으로 사용하다 2015년 공개.
어떤 언어로 되어 있어?
단지 스펙이라서 구현을 하던지 언어별로 구현체가 있으니 가져다 쓰면 됨.http://graphql.org/code/
어디서 쓰고 있어?
Facebook - https://developers.facebook.com/docs/graph-api/using-graph-api)
Github - https://developer.github.com/
GraphQL Query 사용해보기(GraphiQL) : https://developer.github.com/v4/explorer/
Why Github using GraphQL
블로그 : https://githubengineering.com/the-github-graphql-api/
모든 Rest API를 제공하고 있지만 gw(integrators)의 요청을 유연하게 처리하기 힘들었다.
GraphQL을 쓰면 신경 안써도 된다. 스키마만 업데이트 해주면 된다.
때때로 response를 위해 2, 3번의 호출이 필요했다.
GraphQL이 알아서 해준다.
매개 변수에 대한 검증이 필요했다.
매개변수에 대한 검증도 FE의 요청쿼리에 대한 validation 가능하다. eslint-plugin-graphql
코드를 통해 문서를 생성하길 원했다.
GraphiQL을 쓰면 schema에 대한 정보를 볼수 있고 Query도 실행/검증 가능
요청 받을 Response에 대해서 요청자가 정의하길 원한다.
이건 넣고 이건 빼고 할필요 없음. 요청자가 요청한 정보만 내려온다.(=overfetch 방지)
Rest API vs GraphQL HTTP상의 Endpoint 비교
REST API
엔티티별로 아래와 같은 스타일.
리스트 조회 : GET /api/posts
조회 : GET /api/posts/1
생성 : POST /api/posts
치환(수정) : PUT or PATCH /api/posts/1
삭제 : DELETE /api/posts/1
GraphQL
모든 엔티티에 대해서 하나의 URL만 가짐.
POST /graphql
-> 리스트, 조회, 생성, 수정, 삭제등
이정도면...간단하게 Database의 console에 붙어서 Query한다 생각하면 편하다.
다만 Query의 스펙을 미리 정의(=Schema) 해놓아야만 한다. Query 에 따라서 입력/조회/수정/삭제를 한다.
GraphQL Server가 할일은?
1. Node 정의
개별 node를 정의. 예 : 유저, 글, 댓글, 회사, 상품, 판매자
쉽게 이야기하면 Database의 테이블 단위
2. Node 간 관계설정
node에 관계 필드를 추가.- user node에 companyInfo 추가- company node에 users 추가
관계 필드에 대해서 Store 연결.
- companyInfo필드의 Type은 CompanyType- users 필드의 Type은 GraphQLList(UserType)과 같이 감싸준다.
노드(점)의 관계 필드(선)가 곧 Store 연결(=DB 조회)
3. Root Query 생성(선정).
Node중 최상위에서 조회가능한 녀석들을 분류해야한다.
Root Query?
REST API의 여러 조회용 END POINT의 모음이라 생각하면 이해하기 쉽다. 즉. Entry Point를 제공자가 정의해주는 것이다.
RootQuery는 의미적인 표현인 것이고 일반 node와 Type/동작 동일.= GraphQLObjectType
다만 의미적으로 Root Query Node의 경우에는 다른 node를 참조만 하고,다른 노드로 부터 참조되지 않는다.
GraphQL로 서비스 구성 예시
DB든 외부연동서버든 처리할 데이타에 대해서는 Node 정의를 해주어야 합니다.
GraphQL로 서비스 예시 2
여기서도 GraphQL서버에서 Composite을 위해서 서버별 타입별로 Node정의를 해주어야 한다.
만약 A,B 서버에 동일하게 생긴 Node가 있더라도 관계설정이 달라질수 있으므로각각을 구분해주어야 한다.
GraphQL 사용시 주요 장점.
FE/Native의 요청 필드만 리턴하므로 Overfetch 방지.
MSA 구조에서 여러 서버의 데이타를 효율적으로 Data Integration 한다.
REST API에서 2~3회 요청에 걸쳐 데이타를 가져오는 경우를 간단하게 1 회 요청으로 끝낼 수 있다.(유연성 甲)# BE 개발자가 별도 Endpoint 추가하고 로직으로 풀던 문제를 쉽게 끝낼 수 있다.
전체 구조
이렇게도 나누는것도 가능은 할듯.
코드를 보면 이렇게도 가능할 것으로 생각된다.다만 아래와 같이 나눌 경우 동일 데이타에 대해서 변경과 조회가
함께 일어나므로 시점에 따라 값이 달라질수도 있는 문제가 발생 할 수 있다.
샘플코드
회사와 사람 Node
코드로 두 Node로 관계를 지어보자.
샘플을 실행시켜보려면 https://github.com/gidong-lee/graphQL_exam
샘플코드에 대해서.
axios는 back 단 rest api를 호출해주는 jQuery의 ajax같은 라이브러리다.(예 : Spring 개발시 RestTemplate)
JSON Server는 json파일을 읽어 쉽게 개발/목킹용 rest api 서버를 만들어 주는 녀석이다.입력,수정,삭제,페이징등을 모두 처리해준다.GitHub : https://github.com/typicode/json-server
User Type
const UserType = new GraphQLObjectType({
name: 'User', // GraphQL Object Name(필수)
fields: () => ({
id: {type: GraphQLString},
firstName: {type: GraphQLString},
age: {type: GraphQLInt},
companyId: {type: GraphQLString},
company: { /* 연결(user -> company) */
type: CompanyType,
resolve(parentValue, args) {
console.log(parentValue, args);
return axios.get(`${SERVER_DOMAIN}/companies/${parentValue.companyId}`)
.then(req => req.data);
}
}
})
});
위의 fields.company 필드가 관계필드가 된다.
관계 필드는 곧 DB/Cache/rest server 연결(=DB 조회)을 뜻한다.
코드의 axios는 뒤단의 rest api를 호출하여 데이타를 가져오는 것이니 크게신경 안써도 됨.
Company Type
const CompanyType = new GraphQLObjectType({
name: 'Company',
fields: () => ({
id: {type: GraphQLString},
name: {type: GraphQLString},
description: {type: GraphQLString},
users: { /* 연결(company -> users) */
type: new GraphQLList(UserType),
args: {limit: {type: GraphQLInt}},
resolve(parentValue, args) {
const { limit = 20 } = args;
return axios.get(
`${SERVER_DOMAIN}/companies/${parentValue.id}/users?_limit=${limit}`
).then(req => req.data);
}
}
})
});
위의 fields.users 필드가 관계필드가 된다.
관계 필드는 곧 DB/Cache/rest server 연결(=DB 조회)을 뜻한다.
users필드의 타입이 GraphQLList(UserType)인 것을 잘 숙지하자.
Root Query
Query(조회) 시 시작될 수 있는 node 를 지정하여야 한다.따라서 node product 조회는 RootQuery 필드에 없으므로 company를 통해서만 접근 할 수 있다.
RootQuery 지정
const RootQuery = new GraphQLObjectType({
name: 'RootQueryType',
fields: () => ( {
user: { /* RootQueryObject에 User연결 */
type: UserType,
args: {id: {type: GraphQLString}},
resolve(parentValue, args) {
return axios.get(`${SERVER_DOMAIN}/users/${args.id}`)
.then(req => req.data);
}
},
company: { /* RootQueryObject에 Company연결 */
type: CompanyType,
args: {id: {type: GraphQLString}},
resolve(parentValue, args) {
return axios.get(`${SERVER_DOMAIN}/companies/${args.id}`)
.then(req => req.data);
}
}
})
});
관계 필드의 resolve 함수?
parentValue -> 나의 상위에서 주입해준 값.user의 Company에서 조회하여 리턴해준값
args -> Query 호출시에 적어주는 값. 예 : id, searchKeyword, limit, first, last, page, lastNum
const CompanyType = new GraphQLObjectType({
name: 'Company',
fields: () => ({
id: {type: GraphQLString},
name: {type: GraphQLString},
description: {type: GraphQLString},
users: {
type: new GraphQLList(UserType),
args: {limit: {type: GraphQLInt}},
resolve(parentValue, args) {
const { limit = 20 } = args;
return axios.get(
`${SERVER_DOMAIN}/companies/${parentValue.id}/users?_limit=${limit}`
).then(req => req.data);
}
}
})
});
GraphiQL - Graphql의 개발자도구.
작성한 Schema를 보고 Query를 작성하고 유효성을 검사하면서 테스트를 할 수 있는 good dev tool
설치버전도 있고, 서버(dev용으로) 띄울때 함께 띄울수 있다.
Query 해보기
User정보와 회사정보를 가져오는 샘플
요청
{
user(id: "23") {
id
firstName
company {
name
}
}
}
결과
{
"data": {
"user": {
"id": "23",
"firstName": "ko",
"company": {
"name": "tomorrow"
}
}
}
}
Company정보와 users를 가져오는 샘플
요청
{
company(id: "1") {
id
name
description
users {
id
firstName
age
}
}
}
결과
{
"data": {
"company": {
"id": "1",
"name": "tomorrow",
"description": "111st",
"users": [
{
"id": "23",
"firstName": "ko",
"age": 20
},
{
"id": "40",
"firstName": "lee",
"age": 40
},
{
"id": "44",
"firstName": "ra",
"age": 45
}
]
}
}
}
Mutation
변경작업(추가, 수정, 삭제)
Response Type에 따라서 Query를 원하는대로 데이타를 받을 수도 있다.
Mutation 사용시 장점.
사실 로직적으로 내부적으로 변경하는 로직을 있는것 뿐 조회 Query와 구조적으로는 같다.
Mutation query를 호출하더라도 해당 Mutation의 Return Type에 따라서 Response를 자유롭게 변경 할 수 있다.
Mutation - User 추가
const mutation = new GraphQLObjectType({
name: 'Mutation',
fields: {
addUser: {
type: UserType,
args: {
firstName: {type: new GraphQLNonNull(GraphQLString)},
age: {type: new GraphQLNonNull(GraphQLInt)} },
resolve(parentValue, {firstName, age}) {
return axios.post(`${SERVER_DOMAIN}/users`, {firstName, age})
.then(res => res.data);
}
},
deleteUser: {…},
editUser: {…}
}
})
mutation을 이용한 유저 추가 및 생성 정보 받기
요청
mutation {
addUser( firstName: "gu", age: 40 ) {
id
firstName
age
}
}
결과
{
"data": {
"addUser": {
"id": "SJ0Wox-i-",
"firstName": "gu",
"age": 40
}
}
}
Mutation - User 삭제
const mutation = new GraphQLObjectType({
name: 'Mutation',
fields: {
addUser: {…},
deleteUser: {
type: UserType,
args: {
id: {type: new GraphQLNonNull(GraphQLString)}
},
resolve(parentValue, {id}) {
return axios.delete(`${SERVER_DOMAIN}/users/${id}`)
.then(res => res.data);
}
},
editUser: {…}
}
})
mutation을 이용한 유저 삭제
요청
mutation {
deleteUser(id : "SJ0Wox-i-") {
id
}
}
결과
{
"data": {
"deleteUser": {
"id": null
}
}
}
Mutation - User 수정
const mutation = new GraphQLObjectType({
name: 'Mutation',
fields: {
addUser: {…},
deleteUser: {…},
editUser: {
type: UserType,
args: {
id: {type: new GraphQLNonNull(GraphQLString)},
firstName: {type: GraphQLString},
age: {type: GraphQLInt},
companyId: {type: GraphQLString}
},
resolve(parentValue, args) {
return axios.patch(`${SERVER_DOMAIN}/users/${args.id}`, args)
.then(res => res.data);
}
}
}
})
mutation을 이용한 유저 수정(Patch)
요청
mutation {
editUser( id: "44", firstName: "ra", age: 45) {
id,
firstName,
age,
company {
id,
name,
description
}
}
}
결과
{
"data": {
"editUser": {
"id": "44",
"firstName": "ra",
"age": 45,
"company": {
"id": "1",
"name": "tomorrow",
"description": "111st"
}
}
}
}
상황극
GraphQL을 안쓸때 상황.
login.json에 userName이랑 image좀 추가해주세요.
NoticeList.json에서는 id, title, 작성자만 필요하니 나머지는 빼주세요.
몇일있다가 수정일은 추가해주세요.
이 데이타는 결과에 따라서 추가 Request가 필요하므로 서버에서 한번에 말아주세요.
하위 호환 때문에 API 버전 변경이 필요해요.
GraphQL을 쓰게되면.
Root Query에 머하나 추가해주세요.
A 랑 B랑 릴레이션 맺어주심 안돼요?
mutation 추가해주세요.
빨리 끝내고 커피 한 잔 하러 가시죠.(추정)
FrontEnd Client 3총사
Apollo client(http://dev.apollodata.com/)
All in one
다양한 플랫폼 지원(FE(vanilla/react/angular), Native)
relay(https://facebook.github.io/relay/)
FE단 캐싱을 통해 Performance가 좋음.
러닝커브가 있다.
Lokka(https://github.com/kadirahq/lokka)
Simple
URL
slide 예제소스.(https://github.com/gidong-lee/graphQL_exam)
예제쿼리, 소스, 동작방법등 기술.
GraphiQL(https://github.com/graphql/graphiql)
udemy 강의 추천(https://www.udemy.com/graphql-with-react-course)
영어가 힘드시면 저처럼 크롬으로 강의 한글번역해서 보시면 보기 편합니다.
방법 : 크롬으로 강의시작 -> 자막 on(영문) -> 한글로 번역 -> 나쁘지않게 번역됨.
graphql query check eslint(https://github.com/apollographql/eslint-plugin-graphql)
'Javascript/node.js Coding' 카테고리의 다른 글
ES6+(EcmaScript2015+) 정리(ing) (0) | 2017.12.27 |
---|---|
Mobx with angular 정리. (0) | 2017.12.05 |
Netflix Falcor Overview (0) | 2017.12.03 |