Mocking Bird - node.js REST API simulation (1)

當團隊決定開發一個使用 web service 的 mobile app 時,我們遇到了一個小問題:mobile app 跟 web service 是同時開發的,當 web service 還沒實作完畢前,mobile app developer 只能暫時先從規格中 implement 跟 web service 銜接的介面。

這聽起來有點瞎子摸象。

不過 Jamie Sa 提供了個有趣的點子:如果我們可以寫個模擬的 REST API service,並且透過 YAML 定義簡單的規格的話,我們就可以利用這個模擬 REST API 比較真實的跟 web service 銜接了 :)

正巧手邊大家正在研究 node.js,不免俗的我們就用了它來實作了初版的 API Simulator,Jamie 把它叫做 mockingBird。這個時候 YAML spec 看起來像這樣:

\# APIs in V1
meta:
  version: draft
v1\_articles:
  timestamp: '2011-07-19T08:54Z'
  is\_end: true
  articles:
    - { article: 1 }
    - { article: 2 }
    - { article: 3 }
  article\_count: 3

而用瀏覽器開啟 http://127.0.0.1:8080/v1/articles 則會輸出以下結果:

初版的時候我們用 underline "_" 來分割版本,並且將假資料直接放在 YAML 裡面。而當我們需要模擬取得多筆資料的時候還可以處理,但如果需要類似 http://HOST/article/ARTICLE_ID 這樣的 API 就無法模擬了,目前這樣的寫法也無法模擬 POST method。

做了一連串的 hacking 後,終於完成了一個較有彈性的架構,不過當然還是沒辦法模擬太過於複雜的狀況 :P

MockingBird 會將 JSON 格式的假資料讀取進來,而可以在 YAML 裡面調用 dummy data 以及使用簡單的 Javascript 做計算。

比如說我要模擬 Social network 的 API,

首先要先產生 dummy.json 假資料檔,這邊我用一個 python script 產生 dummy.json。

import uuid
import json
from random import randint, choice
from datetime import datetime, timedelta

firstnames = \['Ericka', 'Amie', 'Annabelle', 'Hugh', 'Carmella'\]
lastnames = \['Hilts', 'Kowalsky', 'Cincotta', 'Gerken', 'Stults'\]
devicenames = \['iPad', 'Android', 'Web'\]
basetime = datetime.today()
lipsum='Lorem ipsum dolor sit amet, consectetur adipiscing elit. In sollicitudin elementum tristique. Nullam gravida bibendum magna viverra gravida. Cras nec mi a est malesuada dictum.'

def gen\_id():
    return uuid.uuid1().hex\[:24\]

def gen\_users(num=10):
    users = \[\]
    for i in range(num):
        user = {}
        user\['id'\] = gen\_id()
        user\['email'\] = 'a%[email protected]' % randint(0,1000000)
        user\['nickname'\] = "%s %s" % \\
                           (choice(firstnames), \\
                           choice(lastnames))
        users.append(user)
    return users

def gen\_comment(users, article\_id, timestamp):
    c = {}
    c\['id'\] = gen\_id()
    c\['creator\_id'\] = choice(users)\['id'\]
    c\['creation\_device\_name'\] = choice(devicenames)
    c\['article\_id'\] = article\_id
    c\['timestamp'\] = timestamp
    c\['text'\] = lipsum
    return c

def gen\_articles(users, num=20):
    articles = \[\]
    for i in range(num):
        user = choice(users)
        timestamp = basetime+timedelta(0,i\*10)
        article = {}
        article\['id'\] = gen\_id()
        article\['creator\_id'\] = user\['id'\]
        article\['creation\_device\_name'\] = choice(devicenames)
        article\['timestamp'\] = timestamp.isoformat()
        article\['text'\] = lipsum
        article\['comment\_count'\] = randint(0,5)
        article\['comments'\] = \[\]
        for j in range(article\['comment\_count'\]):
            c = gen\_comment(users, article\['id'\], \\
                            (timestamp+timedelta(0,j\*10)).isoformat())
            article\['comments'\].append(c)

        articles.append(article)
    return articles

if \_\_name\_\_ == '\_\_main\_\_':
    data = {}
    data\['users'\] = gen\_users()
    data\['articles'\] = gen\_articles(data\['users'\])
    print json.dumps(data, indent=2)


這個 dummy json 會 load 進去 node.js 的主程式裡面,可以由程式或者是 YAML 裡面使用。現在的 YAML 檔案則是長成這樣:

version: 0
api:
  articles:
    prefix: api
    url: "articles.\*"
    http\_method: GET
    response:
      timestamp: "timestamp"
      is\_end: "true"
      article\_count: "dummy\['articles'\].length"
      articles: "params\['limit'\] ? dummy\['articles'\].slice(0,params\['limit'\]) : dummy\['articles'\]"
  article:
    prefix: api
    url: "article/(\\w+)$"
    http\_method: GET
    response:
      article: "findById(dummy\['articles'\], match)"
  users:
    prefix: api
    url: "users"
    http\_method: GET
    response:
      users: "dummy\['users'\]"
  
  post\_article:
    prefix: api
    url: article
    http\_method: POST
    response:
      creator\_id: "params.creator\_id"
      creation\_device\_name: "params.creation\_device\_name"
      text: "params.text"
      timestamp: "timestamp"
      comment\_count: "0"
      comments: "\[\]"
      id: "generateId()"
  
  post\_comment:
    prefix: api
    url: comment
    http\_method: POST
    response:
      creator\_id: "params.creator\_id"
      creation\_device\_name: "params.creation\_device\_name"
      article\_id: "params.article\_id"
      text: "params.text"
      id: "generateId()"
      

裡面有些關鍵字解釋一下:

  • version: 最後 mockingBird 的網址會是 http://HOST/PREFIX/VERSION/method,如 http://localhost/api/0/articles,用來區分 API 版本用的變數
  • prefix: 可以針對 API 的用途使用不同的 prefix,在這邊我們只用了 "api" 這個 prefix。
  • url: url 的 match pattern
  • http_method: 可以指定要用 POST 或是 GET 的 HTTP Method
  • response: 要回應的 JSON 訊息。
  • dummy: 在 YAML 裡面可以利用 dummy 取得假資料,比如說 dummy['articles'] 就是所有的 articles。
  • params: 使用 GET/POST 的時候丟進來的參數。比如下達了 http://localhost/api/0/articles?limit=2 時,YAML 可以使用參數 params.limit 取得這個 limit 數值,可以參考下面的片段
  • findById: 用 id 來搜尋物件的 method
articles:
    prefix: api
    url: "articles.\*"
    http\_method: GET
    response:
      timestamp: "timestamp"
      is\_end: "true"
      article\_count: "dummy\['articles'\].length"
      articles: "params\['limit'\] ? dummy\['articles'\].slice(0,params\['limit'\]) : dummy\['articles'\]"

當用瀏覽器開啟 http://localhost:8080/api/0/articles?limit=1 的時候,就可以獲得以下結果:

如此一來,你的 mobile app 就可以開幹了,而有了這個實體的 SPEC,web service 的 developer 也可以依此為目標,最終做成跟這個相同的 API。

下一篇再來詳細講解怎麼使用這個工具還有 node.js 裡面是怎麼實作的。但重點是:有人有興趣嗎?出個聲吧 :P

Yuren 撰於 2011年8月6日