GraphQL 用例:使用 Golang 和 PostgreSQL 構建一個博客引擎 API
摘要
GraphQL 在生產環境中似乎難以使用:雖然對於建模功能來說圖介面非常靈活,但是並不適用於關係型存儲,不管是在實現還是性能方面。
在這篇博客中,我們會設計並實現一個簡單的博客引擎 API,它支持以下功能:
- 三種類型的資源(用戶、博文以及評論)支持多種功能(創建用戶、創建博文、給博文添加評論、關注其它用戶的博文和評論,等等。)
- 使用 PostgreSQL 作為後端數據存儲(選擇它因為它是一個流行的關係型資料庫)。
- 使用 Golang(開發 API 的一個流行語言)實現 API。
我們會比較簡單的 GraphQL 實現和純 REST 替代方案,在一種普通場景(呈現博客文章頁面)下對比它們的實現複雜性和效率。
介紹
GraphQL 是一種 IDL( 介面定義語言 ),設計者定義數據類型和並把數據建模為一個 圖 。每個頂點都是一種數據類型的一個實例,邊代表了節點之間的關係。這種方式非常靈活,能適應任何業務領域。然而,問題是設計過程更加複雜,而且傳統的數據存儲不能很好地映射到圖模型。閱讀附錄1了解更多關於這個問題的詳細信息。
GraphQL 在 2014 年由 Facebook 的工程師團隊首次提出。儘管它的優點和功能非常有趣而且引人注目,但它並沒有得到大規模應用。開發者需要權衡 REST 的設計簡單性、熟悉性、豐富的工具和 GraphQL 不會受限於 CRUD(LCTT 譯註:Create、Read、Update、Delete) 以及網路性能(它優化了往返伺服器的網路)的靈活性。
大部分關於 GraphQL 的教程和指南都跳過了從數據存儲獲取數據以便解決查詢的問題。也就是,如何使用通用目的、流行存儲方案(例如關係型資料庫)為 GraphQL API 設計一個支持高效數據提取的資料庫。
這篇博客介紹構建一個博客引擎 GraphQL API 的流程。它的功能相當複雜。為了和基於 REST 的方法進行比較,它的範圍被限制為一個熟悉的業務領域。
這篇博客的文章結構如下:
- 第一部分我們會設計一個 GraphQL 模式並介紹所使用語言的一些功能。
- 第二部分是 PostgreSQL 資料庫的設計。
- 第三部分介紹了使用 Golang 實現第一部分設計的 GraphQL 模式。
- 第四部分我們以從後端獲取所需數據的角度來比較呈現博客文章頁面的任務。
相關閱讀
- 很棒的 GraphQL 介紹文檔。
- 該項目的完整實現代碼在 github.com/topliceanu/graphql-go-example。
在 GraphQL 中建模一個博客引擎
下述列表1包括了博客引擎 API 的全部模式。它顯示了組成圖的頂點的數據類型。頂點之間的關係,也就是邊,被建模為指定類型的屬性。
type User {
id: ID
email: String!
post(id: ID!): Post
posts: [Post!]!
follower(id: ID!): User
followers: [User!]!
followee(id: ID!): User
followees: [User!]!
}
type Post {
id: ID
user: User!
title: String!
body: String!
comment(id: ID!): Comment
comments: [Comment!]!
}
type Comment {
id: ID
user: User!
post: Post!
title: String
body: String!
}
type Query {
user(id: ID!): User
}
type Mutation {
createUser(email: String!): User
removeUser(id: ID!): Boolean
follow(follower: ID!, followee: ID!): Boolean
unfollow(follower: ID!, followee: ID!): Boolean
createPost(user: ID!, title: String!, body: String!): Post
removePost(id: ID!): Boolean
createComment(user: ID!, post: ID!, title: String!, body: String!): Comment
removeComment(id: ID!): Boolean
}
列表1
模式使用 GraphQL DSL 編寫,它用於定義自定義數據類型,例如 User
、Post
和 Comment
。該語言也提供了一系列原始數據類型,例如 String
、Boolean
和 ID
(它是String
的別名,但是有頂點唯一標識符的額外語義)。
Query
和 Mutation
是語法解析器能識別並用於查詢圖的可選類型。從 GraphQL API 讀取數據等同於遍歷圖。需要提供這樣一個起始頂點;該角色通過 Query
類型來實現。在這種情況中,所有圖的查詢都要從一個由 id user(id:ID!)
指定的用戶開始。對於寫數據,定義了 Mutation
頂點。它提供了一系列操作,建模為能遍歷(並返回)新創建頂點類型的參數化屬性。列表2是這些查詢的一些例子。
頂點屬性能被參數化,也就是能接受參數。在圖遍歷場景中,如果一個博文頂點有多個評論頂點,你可以通過指定 comment(id: ID)
只遍歷其中的一個。所有這些都取決於設計,設計者可以選擇不提供到每個獨立頂點的直接路徑。
!
字元是一個類型後綴,適用於原始類型和用戶定義類型,它有兩種語義:
- 當被用於參數化屬性的參數類型時,表示這個參數是必須的。
- 當被用於一個屬性的返回類型時,表示當頂點被獲取時該屬性不會為空。
- 也可以把它們組合起來,例如
[Comment!]!
表示一個非空 Comment 頂點鏈表,其中[]
、[Comment]
是有效的,但null, [null], [Comment, null]
就不是。
列表2 包括一系列用於博客 API 的 curl
命令,它們會使用 mutation 填充圖然後查詢圖以便獲取數據。要運行它們,按照 topliceanu/graphql-go-example 倉庫中的指令編譯並運行服務。
# 創建用戶 1、2 和 3 的更改。更改和查詢類似,在該情景中我們檢索新創建用戶的 id 和 email。
curl -XPOST http://vm:8080/graphql -d 'mutation {createUser(email:"user1@x.co"){id, email}}'
curl -XPOST http://vm:8080/graphql -d 'mutation {createUser(email:"user2@x.co"){id, email}}'
curl -XPOST http://vm:8080/graphql -d 'mutation {createUser(email:"user3@x.co"){id, email}}'
# 為用戶添加博文的更改。為了和模式匹配我們需要檢索他們的 id,否則會出現錯誤。
curl -XPOST http://vm:8080/graphql -d 'mutation {createPost(user:1,title:"post1",body:"body1"){id}}'
curl -XPOST http://vm:8080/graphql -d 'mutation {createPost(user:1,title:"post2",body:"body2"){id}}'
curl -XPOST http://vm:8080/graphql -d 'mutation {createPost(user:2,title:"post3",body:"body3"){id}}'
# 博文所有評論的更改。`createComment` 需要用戶 id,標題和正文。看列表 1 的模式。
curl -XPOST http://vm:8080/graphql -d 'mutation {createComment(user:2,post:1,title:"comment1",body:"comment1"){id}}'
curl -XPOST http://vm:8080/graphql -d 'mutation {createComment(user:1,post:3,title:"comment2",body:"comment2"){id}}'
curl -XPOST http://vm:8080/graphql -d 'mutation {createComment(user:3,post:3,title:"comment3",body:"comment3"){id}}'
# 讓用戶 3 關注用戶 1 和用戶 2 的更改。注意 `follow` 更改只返回一個布爾值而不需要指定。
curl -XPOST http://vm:8080/graphql -d 'mutation {follow(follower:3, followee:1)}'
curl -XPOST http://vm:8080/graphql -d 'mutation {follow(follower:3, followee:2)}'
# 用戶獲取用戶 1 所有數據的查詢。
curl -XPOST http://vm:8080/graphql -d '{user(id:1)}'
# 用戶獲取用戶 2 和用戶 1 的關注者的查詢。
curl -XPOST http://vm:8080/graphql -d '{user(id:2){followers{id, email}}}'
curl -XPOST http://vm:8080/graphql -d '{user(id:1){followers{id, email}}}'
# 檢測用戶 2 是否被用戶 1 關注的查詢。如果是,檢索用戶 1 的 email,否則返回空。
curl -XPOST http://vm:8080/graphql -d '{user(id:2){follower(id:1){email}}}'
# 返回用戶 3 關注的所有用戶 id 和 email 的查詢。
curl -XPOST http://vm:8080/graphql -d '{user(id:3){followees{id, email}}}'
# 如果用戶 3 被用戶 1 關注,就獲取用戶 3 email 的查詢。
curl -XPOST http://vm:8080/graphql -d '{user(id:1){followee(id:3){email}}}'
# 獲取用戶 1 的第二篇博文的查詢,檢索它的標題和正文。如果博文 2 不是由用戶 1 創建的,就會返回空。
curl -XPOST http://vm:8080/graphql -d '{user(id:1){post(id:2){title,body}}}'
# 獲取用戶 1 的所有博文的所有數據的查詢。
curl -XPOST http://vm:8080/graphql -d '{user(id:1){posts{id,title,body}}}'
# 獲取寫博文 2 用戶的查詢,如果博文 2 是由 用戶 1 撰寫;一個現實語言靈活性的例證。
curl -XPOST http://vm:8080/graphql -d '{user(id:1){post(id:2){user{id,email}}}}'
列表2
通過仔細設計 mutation 和類型屬性,可以實現強大而富有表達力的查詢。
設計 PostgreSQL 資料庫
關係型資料庫的設計,一如以往,由避免數據冗餘的需求驅動。選擇該方式有兩個原因:
- 表明實現 GraphQL API 不需要定製化的資料庫技術或者學習和使用新的設計技巧。
- 表明 GraphQL API 能在現有的資料庫之上創建,更具體地說,最初設計用於 REST 後端甚至傳統的呈現 HTML 站點的伺服器端資料庫。
閱讀 附錄1 了解關於關係型和圖資料庫在構建 GraphQL API 方面的區別。列表3 顯示了用於創建新資料庫的 SQL 命令。資料庫模式和 GraphQL 模式相對應。為了支持 follow/unfollow
更改,需要添加 followers
關係。
CREATE TABLE IF NOT EXISTS users (
id SERIAL PRIMARY KEY,
email VARCHAR(100) NOT NULL
);
CREATE TABLE IF NOT EXISTS posts (
id SERIAL PRIMARY KEY,
user_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE,
title VARCHAR(200) NOT NULL,
body TEXT NOT NULL
);
CREATE TABLE IF NOT EXISTS comments (
id SERIAL PRIMARY KEY,
user_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE,
post_id INTEGER NOT NULL REFERENCES posts(id) ON DELETE CASCADE,
title VARCHAR(200) NOT NULL,
body TEXT NOT NULL
);
CREATE TABLE IF NOT EXISTS followers (
follower_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE,
followee_id INTEGER NOT NULL REFERENCES users(id) ON DELETE CASCADE,
PRIMARY KEY(follower_id, followee_id)
);
列表3
Golang API 實現
本項目使用的用 Go 實現的 GraphQL 語法解析器是 github.com/graphql-go/graphql
。它包括一個查詢解析器,但不包括模式解析器。這要求開發者利用庫提供的結構使用 Go 構建 GraphQL 模式。這和 nodejs 實現 不同,後者提供了一個模式解析器並為數據獲取暴露了鉤子。因此 列表1 中的模式只是作為指導使用,需要轉化為 Golang 代碼。然而,這個「限制」提供了與抽象級別對等的機會,並且了解模式如何和用於檢索數據的圖遍歷模型相關。列表4 顯示了 Comment
頂點類型的實現:
var CommentType = graphql.NewObject(graphql.ObjectConfig{
Name: "Comment",
Fields: graphql.Fields{
"id": &graphql.Field{
Type: graphql.NewNonNull(graphql.ID),
Resolve: func(p graphql.ResolveParams) (interface{}, error) {
if comment, ok := p.Source.(*Comment); ok == true {
return comment.ID, nil
}
return nil, nil
},
},
"title": &graphql.Field{
Type: graphql.NewNonNull(graphql.String),
Resolve: func(p graphql.ResolveParams) (interface{}, error) {
if comment, ok := p.Source.(*Comment); ok == true {
return comment.Title, nil
}
return nil, nil
},
},
"body": &graphql.Field{
Type: graphql.NewNonNull(graphql.ID),
Resolve: func(p graphql.ResolveParams) (interface{}, error) {
if comment, ok := p.Source.(*Comment); ok == true {
return comment.Body, nil
}
return nil, nil
},
},
},
})
func init() {
CommentType.AddFieldConfig("user", &graphql.Field{
Type: UserType,
Resolve: func(p graphql.ResolveParams) (interface{}, error) {
if comment, ok := p.Source.(*Comment); ok == true {
return GetUserByID(comment.UserID)
}
return nil, nil
},
})
CommentType.AddFieldConfig("post", &graphql.Field{
Type: PostType,
Args: graphql.FieldConfigArgument{
"id": &graphql.ArgumentConfig{
Description: "Post ID",
Type: graphql.NewNonNull(graphql.ID),
},
},
Resolve: func(p graphql.ResolveParams) (interface{}, error) {
i := p.Args["id"].(string)
id, err := strconv.Atoi(i)
if err != nil {
return nil, err
}
return GetPostByID(id)
},
})
}
列表4
正如 列表1 中的模式,Comment
類型是靜態定義的一個有三個屬性的結構體:id
、title
和 body
。為了避免循環依賴,動態定義了 user
和 post
兩個其它屬性。
Go 並不適用於這種動態建模,它只支持一些類型檢查,代碼中大部分變數都是 interface{}
類型,在使用之前都需要進行類型斷言。CommentType
是一個 graphql.Object
類型的變數,它的屬性是 graphql.Field
類型。因此,GraphQL DSL 和 Go 中使用的數據結構並沒有直接的轉換。
每個欄位的 resolve
函數暴露了 Source
參數,它是表示遍歷時前一個節點的數據類型頂點。Comment
的所有屬性都有作為 source 的當前 CommentType
頂點。檢索id
、title
和 body
是一個直接屬性訪問,而檢索 user
和 post
要求圖遍歷,也需要資料庫查詢。由於它們非常簡單,這篇文章並沒有介紹這些 SQL 查詢,但在參考文獻部分列出的 github 倉庫中有。
普通場景下和 REST 的對比
在這一部分,我們會展示一個普通的博客文章呈現場景,並比較 REST 和 GraphQL 的實現。關注重點會放在入站/出站請求數量,因為這些是造成頁面呈現延遲的最主要原因。
場景:呈現一個博客文章頁面。它應該包含關於作者(email)、博客文章(標題、正文)、所有評論(標題、正文)以及評論人是否關注博客文章作者的信息。圖1 和 圖2 顯示了客戶端 SPA、API 伺服器以及資料庫之間的交互,一個是 REST API、另一個對應是 GraphQL API。
+------+ +------+ +--------+
|client| |server| |database|
+--+---+ +--+---+ +----+---+
| GET /blogs/:id | |
1. +-------------------------> SELECT * FROM blogs... |
| +--------------------------->
| <---------------------------+
<-------------------------+ |
| | |
| GET /users/:id | |
2. +-------------------------> SELECT * FROM users... |
| +--------------------------->
| <---------------------------+
<-------------------------+ |
| | |
| GET /blogs/:id/comments | |
3. +-------------------------> SELECT * FROM comments... |
| +--------------------------->
| <---------------------------+
<-------------------------+ |
| | |
| GET /users/:id/followers| |
4. +-------------------------> SELECT * FROM followers.. |
| +--------------------------->
| <---------------------------+
<-------------------------+ |
| | |
+ + +
圖1
+------+ +------+ +--------+
|client| |server| |database|
+--+---+ +--+---+ +----+---+
| GET /graphql | |
1. +-------------------------> SELECT * FROM blogs... |
| +--------------------------->
| <---------------------------+
| | |
| | |
| | |
2. | | SELECT * FROM users... |
| +--------------------------->
| <---------------------------+
| | |
| | |
| | |
3. | | SELECT * FROM comments... |
| +--------------------------->
| <---------------------------+
| | |
| | |
| | |
4. | | SELECT * FROM followers.. |
| +--------------------------->
| <---------------------------+
<-------------------------+ |
| | |
+ + +
圖2
列表5 是一條用於獲取所有呈現博文所需數據的簡單 GraphQL 查詢。
{
user(id: 1) {
email
followers
post(id: 1) {
title
body
comments {
id
title
user {
id
email
}
}
}
}
}
列表5
對於這種情況,對資料庫的查詢次數是故意相同的,但是到 API 伺服器的 HTTP 請求已經減少到只有一個。我們認為在這種類型的應用程序中通過互聯網的 HTTP 請求是最昂貴的。
為了利用 GraphQL 的優勢,後端並不需要進行特別設計,從 REST 到 GraphQL 的轉換可以逐步完成。這使得可以測量性能提升和優化。從這一點,API 設計者可以開始優化(潛在的合併) SQL 查詢從而提高性能。緩存的機會在資料庫和 API 級別都大大增加。
SQL 之上的抽象(例如 ORM 層)通常會和 n+1
問題相抵觸。在 REST 示例的步驟 4 中,客戶端可能不得不在單獨的請求中為每個評論的作者請求關注狀態。這是因為在 REST 中沒有標準的方式來表達兩個以上資源之間的關係,而 GraphQL 旨在通過使用嵌套查詢來防止這類問題。這裡我們通過獲取用戶的所有關注者來作弊。我們向客戶提出了如何確定評論並關注了作者的用戶的邏輯。
另一個區別是獲取比客戶端所需更多的數據,以免破壞 REST 資源抽象。這對於用於解析和存儲不需要數據的帶寬消耗和電池壽命非常重要。
總結
GraphQL 是 REST 的一個可用替代方案,因為:
- 儘管設計 API 更加困難,但該過程可以逐步完成。也是由於這個原因,從 REST 轉換到 GraphQL 非常容易,兩個流程可以沒有任何問題地共存。
- 在網路請求方面更加高效,即使是類似本博客中的簡單實現。它還提供了更多查詢優化和結果緩存的機會。
- 在用於解析結果的帶寬消耗和 CPU 周期方面它更加高效,因為它只返回呈現頁面所需的數據。
REST 仍然非常有用,如果:
- 你的 API 非常簡單,只有少量的資源或者資源之間關係簡單。
- 在你的組織中已經在使用 REST API,而且你已經配置好了所有工具,或者你的客戶希望獲取 REST API。
- 你有複雜的 ACL(LCTT 譯註:Access Control List) 策略。在博客例子中,可能的功能是允許用戶良好地控制誰能查看他們的電子郵箱、博客、特定博客的評論、他們關注了誰,等等。優化數據獲取同時檢查複雜的業務規則可能會更加困難。
附錄1:圖資料庫和高效數據存儲
儘管將其應用領域數據想像為一個圖非常直觀,正如這篇博文介紹的那樣,但是支持這種介面的高效數據存儲問題仍然沒有解決。
近年來圖資料庫變得越來越流行。通過將 GraphQL 查詢轉換為特定的圖資料庫查詢語言從而延遲解決請求的複雜性似乎是一種可行的方案。
問題是和關係型資料庫相比,圖並不是一種高效的數據結構。圖中一個頂點可能有到任何其它頂點的連接,訪問模式比較難以預測因此提供了較少的優化機會。
例如緩存的問題,為了快速訪問需要將哪些頂點保存在內存中?通用緩存演算法在圖遍歷場景中可能沒那麼高效。
資料庫分片問題:把資料庫切分為更小、沒有交叉的資料庫並保存到獨立的硬體。在學術上,最小切割的圖劃分問題已經得到了很好的理解,但可能是次優的,而且由於病態的最壞情況可能導致高度不平衡切割。
在關係型資料庫中,數據被建模為記錄(行或者元組)和列,表和資料庫名稱都只是簡單的命名空間。大部分資料庫都是面向行的,意味著每個記錄都是一個連續的內存塊,一個表中的所有記錄在磁碟上一個接一個地整齊地打包(通常按照某個關鍵列排序)。這非常高效,因為這是物理存儲最優的工作方式。HDD 最昂貴的操作是將磁頭移動到磁碟上的另一個扇區,因此最小化此類訪問非常重要。
很有可能如果應用程序對一條特定記錄感興趣,它需要獲取整條記錄,而不僅僅是記錄中的其中一列。也很有可能如果應用程序對一條記錄感興趣,它也會對該記錄周圍的記錄感興趣,例如全表掃描。這兩點使得關係型資料庫相當高效。然而,也是因為這個原因,關係型資料庫的最差使用場景就是總是隨機訪問所有數據。圖資料庫正是如此。
隨著支持更快隨機訪問的 SSD 驅動器的出現,更便宜的內存使得緩存大部分圖資料庫成為可能,更好的優化圖緩存和分區的技術,圖資料庫開始成為可選的存儲解決方案。大部分大公司也使用它:Facebook 有 Social Graph,Google 有 Knowledge Graph。
via: http://alexandrutopliceanu.ro/post/graphql-with-go-and-postgresql
作者:Alexandru Topliceanu 譯者:ictlyh 校對:wxy
本文轉載來自 Linux 中國: https://github.com/Linux-CN/archive