2021-05-27 19:54:00 +09:00
---
marp: true
paginate: true
---
2021-07-19 23:33:51 +09:00
# GraphQL 概論
2021-05-27 19:54:00 +09:00
WebDINO Japan エンジニア
[渡邉浩平 ](https://github.com/kou029w )
![w:200 ](https://github.com/kou029w.png )
2021-07-19 23:33:51 +09:00
<!-- @license https://cdn.jsdelivr.net/npm/highlightjs-graphql @1 .0.2/LICENSE -->
< link rel = "stylesheet" href = "https://cdn.jsdelivr.net/gh/highlightjs/cdn-release @11 .1.0/build/styles/default.min.css" >
< script src = "https://cdn.jsdelivr.net/gh/highlightjs/cdn-release @11 .1.0/build/highlight.min.js" ></ script >
< script > module = { } ; < / script >
< script src = "https://cdn.jsdelivr.net/npm/highlightjs-graphql @1 .0.2/graphql.min.js" ></ script >
< script > hljs . registerLanguage ( "graphql" , hljsDefineGraphQL ) ; hljs . highlightAll ( ) ; < / script >
2021-05-27 19:54:00 +09:00
---
2021-07-19 23:33:51 +09:00
## GraphQL とは
2021-05-27 21:20:20 +09:00
https://graphql.org
---
2021-07-19 23:33:51 +09:00
### GraphQL とは API のクエリ言語
2021-05-27 19:54:00 +09:00
2021-05-28 01:37:08 +09:00
サーバーへの問い合わせ (GraphQL Query)
2021-07-19 23:33:51 +09:00
2021-05-27 19:54:00 +09:00
```graphql
2021-05-28 00:47:16 +09:00
query {
2021-05-27 19:54:00 +09:00
pokemon(name: "Pikachu") {
classification
}
}
```
2021-05-28 01:37:08 +09:00
サーバーからの応答 (JSON)
2021-07-19 23:33:51 +09:00
2021-05-27 19:54:00 +09:00
```json
{
"data": {
"pokemon": {
"classification": "Mouse Pokémon"
}
}
}
```
https://graphql-pokemon2.vercel.app
---
2021-05-29 07:33:09 +09:00
## 歴史
2021-07-19 23:33:51 +09:00
- 2012 年 Facebook による開発
- 2015 年 オープンソース化
- 2019 年 [GraphQL Foundation ](https://graphql.org/foundation/ )に移管
2021-05-29 07:33:09 +09:00
オープンソースな仕様になっており、自由に[貢献可能 ](https://github.com/graphql/graphql-spec/blob/main/CONTRIBUTING.md )
---
## 主な仕様
2021-05-27 19:54:00 +09:00
2021-05-28 01:41:45 +09:00
- [サーバーにデータを要求するための言語 (クエリ言語) ](https://graphql.org/learn/queries/ )
- [クライアントの受け取るデータ形式 (JSON) ](https://graphql.org/learn/serving-over-http/#response )
- [使用可能なデータの構造とその操作を宣言するための言語 (スキーマ言語) ](https://graphql.org/learn/introspection/ )
- [クエリの実行方法 ](https://graphql.org/learn/execution/ )
2021-05-27 19:54:00 +09:00
2021-07-19 23:33:51 +09:00
### GraphQL 以外の身近な言語の例:
2021-05-27 19:54:00 +09:00
- クエリ言語: SQL
- スキーマ言語: [JSON Schema ](https://json-schema.org/ ), [XSD ](http://www.w3.org/TR/xmlschema11-1/ )
---
## 何でないか
- データベースではない
2021-07-19 23:33:51 +09:00
- JavaScript ではない
2021-05-27 19:54:00 +09:00
---
2021-07-19 23:33:51 +09:00
## なぜ GraphQL を使うのか
2021-05-27 19:54:00 +09:00
1. 単一リクエスト
2. 型システム
2021-05-28 01:10:33 +09:00
3. 開発ツール
2021-05-27 19:54:00 +09:00
---
### 1. 単一リクエスト
効率的なデータ読み込み
2021-07-19 23:33:51 +09:00
Facebook が GraphQL を開発した理由は、[モバイルネイティブアプリへの移行のため ](https://reactjs.org/blog/2015/05/01/graphql-introduction.html )
2021-05-27 19:54:00 +09:00
スマホの普及に伴う低速、省電力なデバイスの利用の増加が背景
2021-07-19 23:33:51 +09:00
REST は複数のエンドポイントへの問い合わせを行うという典型的な課題がある
クライアントのデータ取得の効率やデータの表現の都合で似たような振る舞いの API を作りがち
2021-05-28 17:35:45 +09:00
2021-07-19 23:33:51 +09:00
GraphQL は単一のリクエストで指定したデータを取得可能
2021-05-28 17:35:45 +09:00
オーバーフェッチを最小限に抑え、サーバーへのラウンドトリップを少なくする
2021-05-27 19:54:00 +09:00
---
2021-05-27 21:20:20 +09:00
#### REST
2021-05-27 19:54:00 +09:00
2021-05-27 23:18:38 +09:00
![h:600 REST ](https://graphql-engine-cdn.hasura.io/learn-hasura/assets/graphql-react/rest-api.png )
https://hasura.io/learn/graphql/intro-graphql/graphql-vs-rest/
2021-05-27 19:54:00 +09:00
---
2021-05-27 21:20:20 +09:00
#### GraphQL
2021-05-27 19:54:00 +09:00
2021-05-27 23:18:38 +09:00
![h:600 GraphQL ](https://graphql-engine-cdn.hasura.io/learn-hasura/assets/graphql-react/graphql-api.gif )
https://hasura.io/learn/graphql/intro-graphql/graphql-vs-rest/
2021-05-27 19:54:00 +09:00
---
### 2. 型システム
さまざまなフロントエンド環境のサポート
2021-07-19 23:33:51 +09:00
単一 API の構築と正確なデータ構造の維持
2021-05-27 19:54:00 +09:00
クライアントアプリケーションを実行するフロントエンドフレームワークとプラットフォームの多様化が背景
---
2021-05-28 10:07:48 +09:00
#### GraphQL Schema
2021-05-27 21:20:20 +09:00
2021-05-27 19:54:00 +09:00
```graphql=
"""ポケモンを表します"""
type Pokemon {
"""このオブジェクトのID"""
id: ID!
"""このポケモンの全国図鑑No."""
number: String
"""このポケモンの名前"""
name: String
"""このポケモンの重さの最大と最小"""
weight: PokemonDimension
"""このポケモンの高さの最大と最小"""
height: PokemonDimension
"""このポケモンの分類"""
classification: String
# ...
}
```
---
```graphql
"""ポケモンを表します"""
type Pokemon {
}
"""ポケモンの寸法を表します"""
type PokemonDimension {
}
```
オブジェクトの種類
---
```graphql
2021-07-19 23:33:51 +09:00
"""
ポケモンを表します
"""
2021-05-27 19:54:00 +09:00
type Pokemon {
2021-07-19 23:33:51 +09:00
"""
このオブジェクトのID
"""
2021-05-27 19:54:00 +09:00
id: ID!
2021-07-19 23:33:51 +09:00
"""
このポケモンの名前
"""
2021-05-27 19:54:00 +09:00
name: String
2021-07-19 23:33:51 +09:00
"""
このポケモンの分類
"""
2021-05-27 19:54:00 +09:00
classification: String
2021-07-19 23:33:51 +09:00
"""
このポケモンの高さの最大と最小
"""
2021-05-27 19:54:00 +09:00
height: PokemonDimension
}
```
オブジェクトに含まれるフィールド
---
2021-05-27 21:20:20 +09:00
#### 特定のプログラミング言語に依存しない
2021-05-27 19:54:00 +09:00
```javascript
// JavaScript
2021-05-27 21:20:20 +09:00
const pokemonQuery = `{ pokemon(name: "Pikachu") { classification } }` ;
2021-05-27 19:54:00 +09:00
fetch(`http://example/?${new URLSearchParams({ query: pokemonQuery })}` )
2021-07-19 23:33:51 +09:00
.then((r) => r.json())
2021-05-27 19:54:00 +09:00
.then(({ data }) => console.log(data?.pokemon?.classification));
// => "Mouse Pokémon"
```
```kotlin
// Kotlin
val response = apolloClient.query(pokemonQuery).await()
Log.d(response?.data?.pokemon?.classification)
```
```swift
// Swift
apollo.fetch(query: pokemonQuery) { result in
guard let data = try? result.get().data else { return }
print(data.pokemon?.classification)
}
```
https://www.apollographql.com
---
2021-05-28 01:10:33 +09:00
### 3. 開発ツール
2021-05-27 19:54:00 +09:00
短期間での開発
クライアントアプリケーションの設計変更に対応するためのツールが提供されている
継続的デプロイ (CD) が背景
---
2021-05-27 21:20:20 +09:00
#### IDE - GraphiQL
2021-05-27 19:54:00 +09:00
2021-05-27 21:20:20 +09:00
![h:600 GraphiQL ](https://raw.githubusercontent.com/graphql/graphiql/main/packages/graphiql/resources/graphiql.jpg )
2021-05-27 19:54:00 +09:00
2021-05-27 21:20:20 +09:00
https://github.com/graphql/graphiql
2021-05-27 19:54:00 +09:00
---
2021-05-27 21:20:20 +09:00
#### GraphQL Code Generator
2021-05-27 19:54:00 +09:00
コードの生成
```console
$ graphql-codegen
```
使う
```typescript
import { usePokemonQuery } from "./generated";
export default () => {
const { data } = usePokemonQuery();
return data?.pokemon?.classification;
};
```
[React ](https://www.graphql-code-generator.com/docs/plugins/typescript-react-query ), [Vue ](https://www.graphql-code-generator.com/docs/plugins/typescript-vue-apollo ), [Kotlin ](https://www.graphql-code-generator.com/docs/plugins/kotlin ), etc.
2021-05-27 23:13:23 +09:00
https://www.graphql-code-generator.com
2021-05-27 19:54:00 +09:00
---
2021-05-27 23:11:31 +09:00
#### Hasura
2021-07-19 23:33:51 +09:00
GraphQL サーバー
接続したデータベースを自動的に GraphQL API として提供
2021-05-27 23:11:31 +09:00
https://hasura.io
---
2021-05-28 01:10:33 +09:00
## まとめ
2021-05-27 19:54:00 +09:00
2021-07-19 23:33:51 +09:00
- GraphQL とはデータを問い合わせるクエリ言語仕様と周辺技術
2021-05-28 01:10:33 +09:00
- 単一リクエスト/型システム/開発ツール
2021-05-27 19:54:00 +09:00
---
2021-07-19 23:33:51 +09:00
## GraphQL をもっと知る
2021-05-27 19:54:00 +09:00
2021-05-28 17:35:45 +09:00
使うための知識を深める
2021-05-27 19:54:00 +09:00
---
2021-05-28 01:10:33 +09:00
## GraphQL Operation
2021-05-27 19:54:00 +09:00
2021-07-19 23:33:51 +09:00
3 種類の操作
2021-05-28 00:47:16 +09:00
- query - 読み取り
- mutation - 書き込み
- subscription - イベントストリーム
2021-07-19 23:33:51 +09:00
1 つのリクエストに複数の操作を含めることが可能
2021-05-27 19:54:00 +09:00
---
2021-05-28 01:10:33 +09:00
## Query - データの取得
---
2021-05-28 00:47:16 +09:00
### 基本的な構文
2021-05-27 19:54:00 +09:00
2021-05-28 00:47:16 +09:00
```graphql
query {
pokemon(name: "Pikachu") {
classification
}
}
```
`query` … 操作
`name` … 引数
`"Pikachu"` … 値
`pokemon` , `classification` … フィールド
---
子孫関係
```graphql
query {
pokemon(name: "Pikachu") {
classification
height {
minimum
maximum
}
}
}
```
`height` フィールドの中の `minimum` , `maximum` フィールド
---
2021-07-19 23:33:51 +09:00
変数を使った query の再利用
2021-05-28 00:47:16 +09:00
```graphql
query ($name: String!) {
pokemon(name: $name) {
classification
height {
minimum
maximum
}
}
}
```
\+
```json
{
"name": "Pikachu"
}
```
`$name` … 変数
`String!` … 型
---
操作を名付ける
```graphql
query fetchPikachu {
pokemon(name: "Pikachu") {
classification
}
}
```
`fetchPikachu` … 操作名
---
フィールドを名付ける
```graphql
query {
pikachu: pokemon(name: "Pikachu") {
classification
}
}
```
`pikachu` … エイリアス
---
フラグメント
```graphql
fragment dimension on PokemonDimension {
minimum
maximum
}
query {
pokemon(name: "Pikachu") {
classification
height {
...dimension
}
weight {
...dimension
}
}
}
```
`dimension` … フラグメント
---
ディレクティブ
```graphql
query ($showClassification: Boolean!) {
pokemon(name: "Pikachu") {
classification @include (if: $showClassification)
}
}
```
\+
```json
{
"showClassification": true
}
```
`@include` … ディレクティブ
2021-05-27 19:54:00 +09:00
---
2021-05-28 00:47:16 +09:00
## まとめ
2021-07-19 23:33:51 +09:00
- 3 種類の操作
- Query によるデータの取得
2021-05-28 00:47:16 +09:00
---
## 後付
---
### より理解を深めるための知識
2021-07-19 23:33:51 +09:00
- [山本陽平「Web を支える技術」 ](https://gihyo.jp/book/2010/978-4-7741-4204-3 ) - HTTP の基礎知識、REST
2021-05-28 00:47:16 +09:00
- [GraphQL \| A query language for your API ](https://graphql.org/ )
- [How to GraphQL \- The Fullstack Tutorial for GraphQL ](https://www.howtographql.com/ )
- [Fullstack GraphQL Tutorial Series \| Learn GraphQL Frontend and Backend ](https://hasura.io/learn/ )
---
### 何を話していないか
- [Mutation ](https://graphql.org/learn/queries/#mutations )
- [Introspection ](https://graphql.org/learn/introspection/ )
- [エラーレスポンス ](http://spec.graphql.org/June2018/#sec-Errors )
- [Validation ](https://graphql.org/learn/validation/ )
- [Execution ](https://graphql.org/learn/execution/ )
- [Subscription ](https://spec.graphql.org/June2018/#sec-Subscription )
---
2021-05-27 19:54:00 +09:00
### セキュリティ
2021-07-19 23:33:51 +09:00
セキュリティの懸念事項は一般的な Web サービスと同様に存在
2021-05-27 19:54:00 +09:00
- [OWASP Top Ten ](https://owasp.org/www-project-top-ten/ )
- [GraphQL \- OWASP Cheat Sheet Series ](https://cheatsheetseries.owasp.org/cheatsheets/GraphQL_Cheat_Sheet.html )
---
### 認証・認可
2021-07-19 23:33:51 +09:00
GraphQL 仕様に含まないので一般的な Web の認証・認可の設計と同様に行う
2021-05-27 19:54:00 +09:00
- [OpenID ](https://openid.net/ )
- [OAuth ](https://oauth.net/ )
- [JWT ](https://jwt.io/ )
---
### キャッシュ
2021-07-19 23:33:51 +09:00
HTTP GET メソッドによる一般的な[HTTP キャッシュ ](https://developer.mozilla.org/ja/docs/Web/HTTP/Caching )に加え、GraphQL では[グローバルなオブジェクトの識別子の宣言 ](https://graphql.org/learn/global-object-identification/ )による[キャッシュ ](https://graphql.org/learn/caching/ )が存在
2021-05-28 01:10:33 +09:00
---
### JSON Serialization
2021-07-19 23:33:51 +09:00
| GraphQL Value | JSON Value |
| ----------------- | ------------- |
| Map | Object |
| List | Array |
| Null | null |
| String/Enum Value | String |
| Boolean | true or false |
| Int/Float | Number |
2021-05-28 01:10:33 +09:00
https://spec.graphql.org/June2018/#sec -JSON-Serialization