node中間件開發(fā):使用apollo-server和graphQL在KOA項目中搭建graphQL服務(wù)

[TOC]

node中間件開發(fā):使用apollo-server和graphQL在KOA項目中搭建graphQL服務(wù)

中間件作為前端和后端數(shù)據(jù)交互的橋梁,很好地解決了后端修改數(shù)據(jù)結(jié)構(gòu)麻煩和前端對字段結(jié)構(gòu)需求不斷變更之間的矛盾問題。它可以有多種處理方式,本次因為是由前端開發(fā)中間件因此選擇了node作為開發(fā)語言。

graphQL介紹

GraphQL 是一個用于 API 的查詢語言,是一個使用基于類型系統(tǒng)來執(zhí)行查詢的服務(wù)端運行時(類型系統(tǒng)由你的數(shù)據(jù)定義)。GraphQL 并沒有和任何特定數(shù)據(jù)庫或者存儲引擎綁定,而是依靠你現(xiàn)有的代碼和數(shù)據(jù)支撐。

GraphQL 入門

  • GraphQL schema
  • 每一個 GraphQL 服務(wù)都有一個 query 類型,可能有一個 mutation 類型。
  • 標(biāo)量類型(Scalar Types) 它們表示對應(yīng) GraphQL 查詢的葉子節(jié)點
    GraphQL 自帶一組默認(rèn)標(biāo)量類型:
    Int:有符號 32 位整數(shù)。
    Float:有符號雙精度浮點值。
    String:UTF‐8 字符序列。
    Boolean:true 或者 false。
    ID:ID 標(biāo)量類型表示一個唯一標(biāo)識符,通常用以重新獲取對象或者作為緩存中的鍵。ID 類型使用和 String 一樣的方式序列化;然而將其定義為 ID 意味著并不需要人類可讀型。
const recentQueryCarsSchema = buildSchema(`
    type Book {
        title: String
        author: String
    }
    type Query {
        books: [Book],
        hello: String,
        myName: Int,
    }
`);

當(dāng)用作數(shù)據(jù)交互處理時中間件處理流程

image

中間件承擔(dān)了前后端代碼中的一些臟活累活,把兩邊開發(fā)人員都不樂意處理的一些邏輯放在這里面處理。后端只需要輸出數(shù)據(jù),前端只要取到自己想要的數(shù)據(jù)就可以--從這一點上來看中間件就是開了一個新的項目專用于整理后端返回的數(shù)據(jù)。

使用js處理數(shù)據(jù)的優(yōu)缺點

如上文說的,要實現(xiàn)后端數(shù)據(jù)結(jié)構(gòu)的重組,僅僅依靠JavaScript就可以實現(xiàn)了:
在一個新的js項目(暫時叫它 data_handler)中定義好請求的接口和數(shù)據(jù)結(jié)構(gòu),將請求后端得到的數(shù)據(jù)按照定義好的數(shù)據(jù)結(jié)構(gòu)整理一遍再暴露給調(diào)用的項目

在真實的業(yè)務(wù)項目中調(diào)用data_handler暴露出來的方法獲取數(shù)據(jù)

使用js來處理數(shù)據(jù)一大優(yōu)點便是靈活、開發(fā)成本低:說白了就是做一次項目的拆分

而缺點便是拆分出來的項目只適用于原項目,失去了對相同接口的復(fù)用--比如同一個獲取用戶列表的接口,如果另一個項目也需要獲取用戶列表但需要不用的數(shù)據(jù)結(jié)構(gòu),那么當(dāng)前封裝好的方法就不適用了

使用node搭建中間件處理數(shù)據(jù)的優(yōu)缺點

而由node來處理數(shù)據(jù)則完全是在原來的前端項目之外又假設(shè)了一層項目了;前端的請求發(fā)向node層,再由node層做相應(yīng)的處理--向后端請求數(shù)據(jù)或從數(shù)據(jù)庫、緩存中取數(shù)據(jù)返回

使用node進(jìn)行搭建中間件相比于用js單純做數(shù)據(jù)處理優(yōu)點非常明顯:

  • node具有訪問數(shù)據(jù)庫的能力,因此一些簡單的業(yè)務(wù)可以遷移到中間件中進(jìn)行而不用等待后端介入
  • 作為“真正的”中間件它可以整合前后端的請求,比如一個頁面需要請求后端多個接口,那么可以在中間件中將這些接口數(shù)據(jù)整合到一起發(fā)送給前端,減少前端的請求次數(shù)
  • 當(dāng)作為中間件時還可以實現(xiàn)諸如服務(wù)端渲染、頁面緩存、請求緩存之類提升頁面加載速度的功能

但缺點同樣存在:增加了開發(fā)成本,而且同js處理數(shù)據(jù)一般node中間件也需要對不同的項目做特別的處理

引入graphQL讓前端做查詢

可以發(fā)現(xiàn)上文介紹的兩種方法都不能滿足一次編碼前端多個項目對相同接口不同數(shù)據(jù)結(jié)構(gòu)的需求,引入graphQL給予前端查詢、篩選數(shù)據(jù)的能力。服務(wù)端/中間件只要輸出數(shù)據(jù),不用關(guān)心前端需要的數(shù)據(jù)結(jié)構(gòu);而前端只要按照graphQL語法從返回數(shù)據(jù)中查詢出自己需要的數(shù)據(jù)和結(jié)構(gòu),無需關(guān)心后端返回的數(shù)據(jù)結(jié)構(gòu)。

在KOA項目中使用apollo-server和graphQL在KOA項目中搭建graphQL服務(wù)

現(xiàn)有的中間件項目是基于KOA開發(fā)的,因此本次graphQL服務(wù)也是在KOA上搭建。
整體的技術(shù)棧是 KOA + apollo-server + graphql-js

使用KOA就不多說了,因為項目搭建時用的就是KOA。

使用 apollo-server 是因為它為常見的node框架如express、KOA等都實現(xiàn)了graphQL服務(wù);而且還有相應(yīng)的客戶端 apollo-client,支持非常多主流的前端框架如react、VUE等,甚至安卓和IOS客戶端都有相應(yīng)的代碼實現(xiàn),社區(qū)生態(tài)好解決方案完整。具體可以見他們的主頁 Apollo GraphQL

其實一個 apollo-server 就足夠完成graphQL服務(wù)的搭建,但在項目中還額外引入了 GraphQL.js ,原因是使用它來構(gòu)建可讀性更高、適用性更強(qiáng)的GraphQL schema。

While we recommend the use schema-definition language (SDL) for defining a GraphQL schema since we feel it's more human-readable and language-agnostic, Apollo Server can be configured with a GraphQLSchema object ...

代碼實現(xiàn)

項目文件結(jié)構(gòu)大概是

server
└─ server.ts  //入口配置文件
└─ graphQLSchema  //用于定義graphQL的schema及操作方法
└─ model  //實際做的操作

入口文件server.ts中引入 apollo-server-koa 并創(chuàng)建一個 ApolloServer 應(yīng)用到KOA實例中。見官方文檔:

// Construct a schema, using GraphQL schema language
const typeDefs = gql`
  type Query {
    hello: String
  }
`;

// Provide resolver functions for your schema fields
const resolvers = {
  Query: {
    hello: () => 'Hello world!',
  },
};

const server = new ApolloServer({ typeDefs, resolvers });

const app = new Koa();
server.applyMiddleware({ app });

ApolloServer 可以支持不同的傳參構(gòu)建,比如上面代碼傳入 typeDefs, resolvers 由ApolloServer再去構(gòu)建schema:

_apollo-server-core@2.1.0@apollo-server-core\src\apolloserver.ts

apolloserver.ts

makeExecutableSchema方法最終還是調(diào)用了

graphql\type\schema.d.ts

graphql\type\schema.d.ts
graphql\type\schema.d.ts

因此我們也可以直接傳入一個schema,文檔中同樣有說明:

const { ApolloServer, gql } = require('apollo-server');
const { GraphQLSchema, GraphQLObjectType, GraphQLString } = require('graphql');

// The GraphQL schema
const schema = new GraphQLSchema({
  query: new GraphQLObjectType({
    name: 'Query',
    fields: {
      hello: {
        type: GraphQLString,
        description: 'A simple type for getting started!',
        resolve: () => 'world',
      },
    },
  }),
});

const server = new ApolloServer({ schema });

本次開發(fā)中采用了第二種直接傳入schema的方式,做了一些模塊的拆分:

server.ts

// graphQL server
import { ApolloServer } from 'apollo-server-koa';
// graphQL schema
import schema from './graphQLSchema'

const app = new Koa();
const router = new Router();

// 生成一個graphQL服務(wù)并應(yīng)用到KOA中
const apolloServer = new ApolloServer({
  schema
});
apolloServer.applyMiddleware({ app });

server\graphQLSchema\index.ts 構(gòu)建ApolloServer的schema

'use strict';
import {
  GraphQLObjectType,
  GraphQLSchema,
} from 'graphql';
import { sth } from './childModule';


// graphQL的入口文件,引入各個查詢的對象
const schema = new GraphQLSchema({
  query: new GraphQLObjectType({
    name: 'Query',
    fields: {
      sth,
    },
  })
});

export default schema;

server\graphQLSchema\childModule\index.ts 查詢返回數(shù)據(jù)的類型聲明及調(diào)用的方法

'use strict';
import { GraphQLList, GraphQLString } from 'graphql';
import { childModuleSchema} from './schema';
import { getChildModuleData} from '../../model/childModule';

export const recentQueryCarsList = {
    type: new GraphQLList(childModuleSchema),
    args: {
        token: { type: GraphQLString }
    },
    async resolve(source: any, args: any) {
        const token = args.token || '';
        if (!token) {
            return [];
        }
        return await getChildModuleData(token);
    }
}

server\graphQLSchema\childModule\schema.ts 聲明返回數(shù)據(jù)具體的結(jié)構(gòu)

'use strict';
import { GraphQLString, GraphQLObjectType, GraphQLInt } from 'graphql';

export const childModuleSchema= new GraphQLObjectType({
    name: 'childModuleSchema',
    description: 'childModuleSchema detail item',
    fields: () => ({
        name: {
            type: GraphQLString,
        },
        age: {
            type: GraphQLString,
        },
        height: {
            type: GraphQLInt,
        }
    })
});

通過以上文件搭建一個graphQL的服務(wù),接收客戶端的請求并將其轉(zhuǎn)發(fā)到服務(wù)端,將服務(wù)端返回的數(shù)據(jù)以graphQL的服務(wù)形式提供給客戶端做查詢。當(dāng)然也可以從數(shù)據(jù)庫或其它地方獲取到數(shù)據(jù),在getChildModuleData方法中做相應(yīng)處理即可。
時間倉促僅能跑起一個查詢的小demo,代碼中有紕漏還請大家指正。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,363評論 6 532
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,497評論 3 416
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,305評論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,962評論 1 311
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 71,727評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,193評論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,257評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,411評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,945評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,777評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,978評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,519評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,216評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,642評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,878評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,657評論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 47,960評論 2 373