10
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

新型コロナウイルス対策サイトの data.json を外部から読み込めるよう疎結合化してみた

Last updated at Posted at 2020-03-24

開発した動機

新型コロナウイルス対策サイトを1からブラウザだけで更新データを編集してプレビューまでできるようにしてみた

この作者さんの思いを感じ、

「たしかに、データとプログラム部分は分離されているほうが便利なことも多いよなぁ」

とおもったので、疎結合にしてみました。

※ 完全自動化されて、データの取り込みなんて、1日1回のバッチ処理だぜぃ! な東京都やオープンデータ完備な自治体対応の場合は、現状の静的サイト生成も大正解だと思います。

結論

data.json の一部を長野県専用のレポジトリから動的に読み込み

GitHub にて、この commit 差分を眺めてもらうだけでも、分かる人にはわかるかなと思いますが、すこし解説を Qiita に残そうと思いました。

ちょっと Bad Hack 感ものこるので、もしそのあたりで、もっとベストプラクティスあれば、コメント欄などで教えてやってくださいませ。

手順解説

Store をつかおう

新型コロナウウイルス対策サイトは、NuxtJS というフレームワークを用いてつくられています。

いくつかやりかたはあるのですが、NuxtJS (Vuex) が機能として持っている、Store にデータを取り込むことにしました。

Store は、Reactive なデータ更新を前提とした機能ですが、今回はそのような使い方はしないで、Webアプリ起動時の一回きりの読み込みとしています。

... Qiita に、Github の diff とか、かんたんに埋め込む方法、ないのかしら

/store/data.js をつくりましょう。

ここでしていることは簡単で、Store のデータ領域を用意し、そこに axios で、外部サイトから data.json を読み込んでくる Action をつくっています。

なお、わたしが携わっている長野県版の場合は、まだ完全な外部JSON処理が固まっていないため、初期値は、/data/data.json を読み込み、外部サイトからの data.json 読み込みは、その一部を上書きするような処理となっています。

Middleware をつくろう

Store に、データを読み込まなければいけません。

ここがすこし Bad Hack 感あるので、もっといい方法をご存じの方がいれば教えていただきたいのですが、今回は、NuxtJS の Middleware という機能をつかって、Webアプリ起動時に、一度だけ data.json を読み込むようにしました。

/middleware/getData.js をつくりましょう。

store.state.data.data が空の場合に、先程作った、Store の Action をキックする処理です。

Middleware は、NuxtJS のアプリ内でページ遷移が発生するたびにキックされるような仕組みです。

このように記述することで、アプリ起動時のみ、Store の Action がキックされるようになります。

そうそう、つくった middleware を有効にするように、nuxt.config.js にも追記が必要です。

URLを環境変数に

nuxt.config.js をさわるついでに、data.json を取得してくる外部URLを、環境変数化してしまいましょう。

※ このURLにアクセスする際には、access-control-allow-origin が適切に返される必要があります。それについて、わからないかたは、別途ググってくださいね。

data.json が使われているところを store に置き換える

とにかく

import Data from '@/data/data.json'

となっているソースコードを検索して、置き換えました。

基本的に、

// import Data from '@/data/data.json'

この行を削除するなりコメントアウトするなりし、vue のクラス内の data() 直後に、一行いれるだけで対応するはずです。

  data() {
    const Data = this.$store.state.data.data

ちょっとまった!!

ここまでの方法だと、spa モードでしたうまくうごいてませんでした。
Bad Hack 感があったの場所が、やはり考慮不足でした。反省。。。再度対応しました。

お手数かけますが、こちらの Branch も参照ください。

data.json の読み込みが universal モードではうまく動いてなかったので修正

middleware で、「Webアプリ起動時のみ一回だけ axios で json 読み込み」は、

  • Universal モードでは、SSRで読み込まれ、クライアントサイドでルートが変更されるときに読み込まれる

  • spa モードでは、毎回読み込まれる

という挙動で、今回のように netlify などに静的HTMLとして公開しているときは悪手でした。

なので、

nuxt-client-init-module

のお力を借りることにしました。

$ yarn add nuxt-client-init-module

nuxt.config.json に、そのモジュールを利用するよう追記

先に追加した middleware 関係を削除します。

$ rm -f ./middleware

nuxt.config.js からも middlewareを削除

/store/index.js をつくります

export const actions = {
  async nuxtClientInit({ dispatch }) {
    // code
    await dispatch('data/getDataAction')
  }
}

ややこしくしちゃって、ごめんなさい。

最後に

ほかにも、news.json や metro.json など、外部読み込み化したい JSON もありますが、同じようなやりかたで実装できると思います。

長野県版でも、そのあたりのデータ更新について、オープンデータからバッチ処理で自動化できたり、人間手作業担当やその手順が決まったりしたら、実装すると思います。

10
3
1

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
10
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?