Web RecordWEB Record
Firebase

【Firebase】Firebase Emulatorの導入方法

2024.06.16
【Firebase】Firebase Emulatorの導入方法

Firebaseからは、Firebase Emulatorという機能が提供されています。

これはFirebaseの各種サービス(Cloud FirestoreやFirestorage、Authなど)の挙動をローカル環境でシュミレートできるものです。

エミュレーターを使用するメリットとして、下記が挙げられます。

Firebase Emulatorを使用するメリット
  • ローカル環境で実際のFirebaseに接続することなくFirebaseの機能を使用できる。
  • 意図せず課金が高くなってしまうような処理をローカル環境で発見できる(エミュレータなので料金がかからない)
  • テストの自動化を行いやすい

本記事では、Firebase Emulatorの導入方法を説明していきます。

firebase-toolsをインストールし初期化する。

firebase-toolsのインストール

Firebase Emulatorはfirebase-toolsをインストールすることで利用できるようになります。

これは、FirebaseをCLI上で操作できるようになるツールです。

まずは、下記コマンドでインストールを行います。

bash
1npm i firebase-tools -g
注意

firebase-toolsをインストールする際に、CLIスクリプトを利用したcurl -sL firebase.tools | bashがFirebaseのドキュメントに記載されている場合があります。

この方法でインストールすると内部的にはNodeもインストールされるようです。
ただ、CLIスクリプトでインストールしたNodeのバージョンだと、エミュレーター起動時にエラーになります。

必ず npm i firebase-tools -g のnpm経由でインストールするように注意してください。

上記問題に関する情報は下記のissueに記載があります。

https://github.com/firebase/firebase-tools/issues/6931

グローバル環境にインストールしたくない場合は、適宜オプションを変更してください。

例えば、下記です。

bash
1npm i firebase-tools -D

インストールされたことを確認します。

bash
1npx firebase-tools --version
2
313.11.2
4
5npx firebase --version
6
713.11.2

バージョン情報が確認できればOKです。

firebaseの初期化

次にfirebaseをCLI上から操作できるように初期化を行います。

bash
1firebase init

何点か質問がくるので答えていきます。

私の場合は、下記で選択しました。

bash
11. Which Firebase features do you want to set up for this directory? Press Space to select features, then Enter to confirm your choices.
2
3どのサービスを使用するかを選択します。
4今回はエミュレーターを使用したいので Emulators: Set up local emulators for Firebase products を選択します。
5
62. Please select an option
7
8接続するFirebaseのプロジェクトの種類を選択します。
9選択肢は下記です。
10
11Use an existing project
12Create a new project
13Add Firebase to an existing Google Cloud Platform project
14Don't set up a default project
15
16既存のプロジェクトを使いたかったので、 Use an existing project を選択しました。
17
183. Select a default Firebase project for this directory
19
20接続するFirebaseのプロジェクトを選択します。
21
224. Which Firebase emulators do you want to set up? Press Space to select emulators, then Enter to confirm your choices.
23
24どの機能をエミュレートするかを選択します。(複数選択可)
25今回は Cloud Firestore をエミュレートしたかったので、Firestore Emulator を選択しました。
26
275. Which port do you want to use for the firestore emulator? (8080)
28
29エミュレートする機能のポートを設定します。デフォルトの8080を設定しました。
30
316. Would you like to enable the Emulator UI? (Y/n)
32
33Emulator UI を使用するかどうかを選択します。
34Emulator UI とはエミュレーターの各種データの確認などをブラウザ上から確認できる機能です。
35今回は使用したかったので「Y」を選択しました。
36
377. Which port do you want to use for the Emulator UI (leave empty to use any available port)?
38
39エミュレーターUIにアクセスする際のポートを設定します。デフォルトは4000になります。
40デフォルトで問題なければそのままエンターキーを押します。
41
428. Would you like to download the emulators now?
43
44エミュレーターをすぐにダウンロードするかを設定します。 Y を選択。
45

質問に答えると、初期化されてエミュレーターを使用できるようになります。

下記のようにコンソールに表示されればOK。

bash
1i  Writing configuration info to firebase.json...
2i  Writing project information to .firebaserc...
3i  Writing gitignore file to .gitignore...
4
5✔  Firebase initialization complete!

初期化後は以下のファイルが生成されます。

.firebaserc
1{
2  "projects": {
3    "default": "xxx"
4  }
5}
firebase.json
1{
2  "emulators": {
3    "firestore": {
4      "port": 8080
5    },
6    "ui": {
7      "enabled": true
8    },
9    "singleProjectMode": true
10  }
11}
.gitignore
1# Logs
2logs
3*.log
4npm-debug.log*
5yarn-debug.log*
6yarn-error.log*
7firebase-debug.log*
8firebase-debug.*.log*
9
10# Firebase cache
11.firebase/
12
13# Firebase config
14
15# Uncomment this if you'd like others to create their own Firebase project.
16# For a team working on the same Firebase project(s), it is recommended to leave
17# it commented so all members can deploy to the same project(s) in .firebaserc.
18# .firebaserc
19
20# Runtime data
21pids
22*.pid
23*.seed
24*.pid.lock
25
26# Directory for instrumented libs generated by jscoverage/JSCover
27lib-cov
28
29# Coverage directory used by tools like istanbul
30coverage
31
32# nyc test coverage
33.nyc_output
34
35# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
36.grunt
37
38# Bower dependency directory (https://bower.io/)
39bower_components
40
41# node-waf configuration
42.lock-wscript
43
44# Compiled binary addons (http://nodejs.org/api/addons.html)
45build/Release
46
47# Dependency directories
48node_modules/
49
50# Optional npm cache directory
51.npm
52
53# Optional eslint cache
54.eslintcache
55
56# Optional REPL history
57.node_repl_history
58
59# Output of 'npm pack'
60*.tgz
61
62# Yarn Integrity file
63.yarn-integrity
64
65# dotenv environment variables file
66.env
67

エミュレーターを起動する。

初期化が完了したら、エミュレーターを起動します。

bash
1npx firebase emulators:start
2
3i  emulators: Starting emulators: firestore
4⚠  firestore: Cloud Firestore Emulator does not support multiple databases yet.
5⚠  firestore: Did not find a Cloud Firestore rules file specified in a firebase.json config file.
6⚠  firestore: The emulator will default to allowing all reads and writes. Learn more about this option: https://firebase.google.com/docs/emulator-suite/install_and_configure#security_rules_configuration.
7i  firestore: Firestore Emulator logging to firestore-debug.log
8✔  firestore: Firestore Emulator UI websocket is running on 9150.
9i  ui: Emulator UI logging to ui-debug.log
10
11┌─────────────────────────────────────────────────────────────┐
12│ ✔  All emulators ready! It is now safe to connect your app. │
13│ i  View Emulator UI at http://127.0.0.1:4000/               │
14└─────────────────────────────────────────────────────────────┘
15
16┌───────────┬────────────────┬─────────────────────────────────┐
17│ Emulator  │ Host:Port      │ View in Emulator UI             │
18├───────────┼────────────────┼─────────────────────────────────┤
19│ Firestore │ 127.0.0.1:8080 │ http://127.0.0.1:4000/firestore │
20└───────────┴────────────────┴─────────────────────────────────┘
21  Emulator Hub running at 127.0.0.1:4400
22  Other reserved ports: 4500, 9150
23
24Issues? Report them at https://github.com/firebase/firebase-tools/issues and attach the *-debug.log files.

この状態で、http://localhost:4000/にアクセスできればうまくいっています。

今回はColud Firestoreをエミュレートしたので、上記のFirestore emulatorから入ることができます。

もちろんデータを生成することも可能です。

エミュレーター内のデータの永続化

今回は、Cloud Firestoreのエミュレーターを使用しましたが、生成したデータはエミュレーターを閉じるとリセットされてしまいます。

エミュレーター起動時に下記のオプションを設定することでこの問題を回避できます。

bash
1npx firebase emulators:start --import data --export-on-exit

すると下記のようにdataディレクトリにエミュレーター内のデータが保存されます。

次回起動時にもnpx firebase emulators:start --import data --export-on-exitとコマンドを打って起動すれば前回のデータにアクセスすることができます。

ただ毎回npx firebase emulators:start --import data --export-on-exitと打つのは大変なので、package.jsonにコマンドを登録しておくと便利です。

package.json
1{
2  ...
3  "scripts": {
4    "dev:firestore": "npx firebase emulators:start --import data --export-on-exit"
5    ...
6  },
7  ...
8}

ローカル環境時にエミュレーターに接続させる

ここまでで、エミュレーターを起動されることができました。

最後に、ローカル環境で開発しているときはエミュレーターに接続するようにすれば完了です。

Firebaseの各機能ごとにエミュレーターに接続する関数がSDKから公開されているのでローカル環境時にはそれを使用するようにします。

SDKをインストールしていない場合は下記でインストールしておいてください。(javascriptの例になります。)

bash
1npm install firebase

今回は、Cloud Firestoreのエミュレーターを使用しているのでそれ用の関数を使用することになります。

使用する関数はconnectFirestoreEmulatorになります。

connectFirestoreEmulatorのインターフェースは下記になります。

connectFirestoreEmulator

connectFirestoreEmulator(Firestore, エミュレーター接続ホスト(文字列), エミュレーター接続ポート(数字))

接続ホストと接続ポートはfirebase.jsonに記載されているhostpostの値になります。

今回は下記のようになっているので、ホストは'127.0.0.1'でポートは8080になります。

firebase.json
1{
2  ...
3  "emulators": {
4    "firestore": {
5      "port": 8080,
6      "host": "127.0.0.1"
7    },
8    ...
9  }
10}

下記がjavascriptSDKを利用してローカル環境時にエミュレーターに接続する例になります。

この例では、どの環境で動いているかの情報を環境変数に格納してその値をみて、ローカル環境か本番環境かを判定します。

ローカル環境の場合

.env(ローカル)
1ENVIRONMENT="Local"

本番環境の場合

.env(本番)
1ENVIRONMENT="Production"

以下はローカル環境下ではエミュレーターに接続するコード例になります。

js
1import { initializeApp } from 'firebase/app';
2import { collection, connectFirestoreEmulator, getDocs, getFirestore } from "firebase/firestore";
3
4const ENVIRONMENT = process.env.ENVIRONMENT; // 環境変数からどの環境かの情報を取得する。ローカル環境の場合はLocalの値が入る
5
6// Firebaseの設定
7const firebaseConfig = {
8  apiKey: process.env.FIREBASE_API_KEY,
9  authDomain: process.env.FIREBASE_AUTH_DOMAIN,
10  projectId: process.env.FIREBASE_PROJECT_ID,
11  storageBucket: process.env.FIREBASE_STORAGE_BUCKET,
12  messagingSenderId: process.env.FIREBASE_MESSAGING_SENDER_ID,
13  appId: process.env.FIREBASE_APP_ID,
14};
15const app = initializeApp(firebaseConfig);
16
17// Firestoreの取得
18const db = getFirestore(app);
19
20if (process.env.ENVIRONMENT === "Local") {
21   // ローカル環境の場合にエミュレーターに接続する
22  connectFirestoreEmulator(db, '127.0.0.1', 8080);
23}
24
25const articles = collection(db, 'articles');
26const articlesSnapshot = await getDocs(articles);
27const result = articlesSnapshot.docs.map((a) => {
28  return a.data();
29})
30
31console.log(result);

エミュレーターでは下記のデータが登録されています。

この状態で、console.logの部分でエミュレーターのデータが取得できていればOKです。

console.logで取得できたデータは下記になります。

bash
1[
2  {
3    createdAt: Timestamp { seconds: 1718534774, nanoseconds: 790000000 },
4    userType: 'guest',
5    url: 'https://example.com',
6    title: 'tset!!!!!!!!!!!'
7  }
8]

ちゃんとエミュレーターのデータが取得できました。