なんでやろうと思ったのか
会社で基本設計書といえば、ExcelとかWordとかPPTとか、Office系のアプリを使って作成することが多い。その場合、下記の点でいつも悩ましい状態になる。
- バージョン管理
- バージョン間の差分管理
- レビュー指摘票と行数のマッピングのズレ(指摘が複数ある場合に行数や連番がずれて、どこの指摘の修正がどこに行ったのかわからなくなる)
- レビューのクローズ状況の追跡
(最終的には画像化してPPTに貼っているけど。。。)
なので、上記の悩ましい状態の解決方法として、最近慣れてきているMarkdownでの基本設計書の作成をしていきたいと考えていた。
ということで、試してみたので、その備忘録です。
で、どうだった?
記述ルールは簡単で、モックサーバをローカルに立てみて、- 設計と実装意図があっているか確認できる
- 実コードが無くてもローカルでAPIを利用して、利用者側の実装ができそう
HTML出力したレイアウトも結構きれいで、好みでした。
ただし、レビュー指摘票の行数のマッピングのズレについては、GithubのPullRequestを利用しないと解決できないので、今後の課題ということで。。。
あれ?VSCodeに入れたExtentionを使っていない。
ま、いっか〜
実装内容の公開
今回作成したMarkdownのファイルは下記のGithubで公開します。不備などありましたら、ご指摘いただけますと幸いです。
o310yusuke/APIBlueprint_Tutorial
何を使うのか
- テキストエディタ:VSCode
- 理由:使い慣れてきたし、軽いし。
- エディタ拡張機能:vscode-apielements
- 理由:公式サイトapiblueprintに乗っていたから
- 設計書記述方法:API Blueprint
- 理由:参考サイトを参考に(記述ルールがシンプルそうだったから)
- レンダリング:Aglio
- 理由:参考サイトを参考に(node.jsでnpmで導入できるから)
- モックサーバ:Drakov
- 理由:参考サイトを参考に(node.jsでnpmで導入できるから)
参考サイト
- API仕様書はMarkdownで書いてhtml出力がベスト(API Blueprint) by タカフ@柏のWeb屋さん
- 「webapi markdown」とGoogle先生に質問したときに見つけたサイトで、本記事のメインの参考サイト
- 開発効率を上げるWebAPIドキュメントの書き方
- 上記のサイトからリンクされていたサイト
- API Blueprintの詳しい書き方
- apiblueprint公式サイト(もちろん英語)
- aglioでAPI Blueprintを使ったドキュメント作成環境をローカルに構築する
- npm installするときはどのパッケージもローカルインストールで十分
- グローバルインストールしなかったら、コマンドが使えなかったので、調べた結果
やったこと(環境構築)
実施したOSはMacです。Node.jsは7.10.0です(古かった。。。)。
VSCodeに拡張機能をインストール
- apielementで検索してインストール
- workspaceとして、「apiblueprint_trial」のディレクトリ作成
npmのインストール状況を確認
$ npm ls --depth=0 -g
/usr/local/lib
├── generator-code@1.2.0
├── npm@4.2.0
└── yo@3.1.0
generator-codeとyoは、VSCodeのExtensionを作ってみたくていらたものなので、関係無し。Aglioをインストール
参考サイトではグローバルにインストールしていたけど、今回はローカルのみにインストールを行う。# node.jsのグローバルにインストールする場合
npm install -g aglio
# 特定ディレクトリにインストールする場合
cd 特定ディレクトリまで移動
npm install aglio
# インストール確認(直下のものだけ)
npm ls --depth=0
npm ls -g --depth=0
結果$ npm ls --depth=0
/apiblueprint_trial
└── aglio@2.3.0
$ npm ls --depth=0 -g
/usr/local/lib
├── generator-code@1.2.0
├── npm@4.2.0
└── yo@3.1.0
Drakovをインストール
こちらもローカルのみにインストールnpm install drakov
結果$ npm ls --depth=0
/apiblueprint_trial
├── aglio@2.3.0
└── drakov@1.0.4
$ npm ls --depth=0 -g
/usr/local/lib
├── generator-code@1.2.0
├── npm@4.2.0
└── yo@3.1.0
やったこと
API Blueprintの公式サイトにチュートリアルがあったので、そちらを参考に実施。Markdownで書いてみる
- tutorial.mdを作成する。
- まず、API名とメタデータを決める。
FORMAT: 1A ←Blueprintのバージョン # Polls ←API名 サンプルAPIの名前を「Polls」にします。 ←APIの説明
- リソースグループを「# Group」で記載する。
- チュートリアルでは、「質問」に関する複数のAPIの章立てのイメージみたい。
# Group 質問 ←グループ名 「質問」に関するAPIのリソースを集約する。 ←グループの説明
- リリースを定義する。
## 質問コレクション [/questions] ←「質問」のリソースグループ内の質問のリスト。[]内にURLテンプレートを記述可能。
- アクションを定義する。
- 1つのリソースに必ず1つのアクションを定義すること。
※チュートリアルでは、2つめのアクションを記載しているが割愛### 一覧取得 [GET] ←リソース内の1つのメソッド。[]内にHTTPメソッドを記載する。 - Response 200 (application/json) [ { "question": "Favourite programming language?", "published_at": "2014-11-11T08:40:51.620Z", "url": "/questions/1", "choices": [ { "choice": "Swift", "url": "/questions/1/choices/1", "votes": 2048 },{ "choice": "Python", "url": "/questions/1/choices/2", "votes": 1024 } ] } ]
※チュートリアルでは、2つめのリソースを記載しているが割愛
Markdown→HTML変換してみる
- 作成したtutorial.mdから、aglioを使ってHTMLを作成する
- aglioをローカルインストールしたので、下記のコマンドでは実行できなかった。
$ aglio -o tutorial.md -o tutorial.html
- 下記のコマンドを実行し、npxをグローバルインストールし、npmの初期化を実施した(デフォルト値で実行した)。
$ npm install -g npx # 結果 $ npm ls --depth=0 -g /usr/local/lib ├── generator-code@1.2.0 ├── npm@4.2.0 ├── npx@10.2.0 └── yo@3.1.0 # 初期化 $ npm init
- npxを利用することで、ローカルインストールしたパッケージも利用可能にできた。
$ npx aglio -i tutorial.md -o tutorial.html
- 表記のエラーがあると、コンパイルエラーとなり、修正方法も提示してくれる。それに合わせて修正することで、コンパイルしてくれる。
- aglioをローカルインストールしたので、下記のコマンドでは実行できなかった。
- 下記コマンドを実行しておけば、mdファイルを保存したタイミングでリアルタイムで変換してくれる。
npx aglio -i tutorial.md --server # 結果 Server started on http://127.0.0.1:3000/ Rendering tutorial.md # 終了方法 Ctrl + C
Mockサーバを試してみる
- drakovでMockサーバを起動する。
npx drakov -f tutorial.md --watch # 結果 [INFO] No configuration files found [INFO] Loading configuration from CLI DRAKOV STARTED [LOG] Setup Route: GET /questions 一覧取得 [LOG] Setup Route: POST /questions 新規投稿 [LOG] Setup Route: GET /questions/:question_id 質問詳細取得 [LOG] Setup Route: DELETE /questions/:question_id 削除 Drakov 1.0.4 Listening on port 3000 FILE SPY ACTIVE # 終了方法 Ctrl + C
- ブラウザで、一覧を取得してみる。
- URL: http://localhost:3000/questions
- 結果(コンソール)
[LOG] GET /questions [MATCHING] by url pattern: /questions/:question_id NOT_MATCHED [MATCHING] by url pattern: /questions MATCHED [DRAKOV] GET /questions 一覧取得
- 結果(ブラウザ)
[ { "question": "好きな開発言語はなんですか?", "published_at": "2014-11-11T08:40:51.620Z", "url": "/questions/1", "choices": [ { "choice": "Swift", "url": "/questions/1/choices/1", "votes": 2048 },{ "choice": "Python", "url": "/questions/1/choices/2", "votes": 1024 } ] } ]
- cURLを使った確認
- 入力画面
# 一覧取得(HeaderとBodyを表示するため「-i」を付与) $ curl -i http://localhost:3000/questions # 新規登録 $ curl -i -X POST -H "Content-type: application/json" -d '{"question":"好きな開発言語はなんですか?","choices":["Swift","Python"]}' http://localhost:3000/questions # 削除 curl -i -X DELETE http://localhost:3000/questions/1
- ウォッチ画面(Drakovを起動したコマンドプロンプト)
# 一覧取得 [LOG] GET /questions [MATCHING] by url pattern: /questions/:question_id NOT_MATCHED [MATCHING] by url pattern: /questions MATCHED [DRAKOV] GET /questions 一覧取得 # 新規登録 [LOG] POST /questions [MATCHING] by url pattern: /questions/:question_id NOT_MATCHED [MATCHING] by url pattern: /questions MATCHED [MATCHING] by request content type: application/json actual: application/json MATCHED [MATCHING] by request body content MATCHED [DRAKOV] POST /questions 新規投稿 # 削除 [LOG] DELETE /questions/1 [MATCHING] by url pattern: /questions/:question_id MATCHED [DRAKOV] DELETE /questions/{question_id} 削除
- 入力画面