Đồng bộ hóa Firebase Realtime Database trong Laravel
Trong quá trình sử dụng Firebase, sẽ có lúc bạn cần đồng bộ dữ liệu của mình lên firebase mỗi khi có sự thay đổi. Bạn muốn mỗi khi có sự thay đổi ở bảng users trên DB của mình thì sự thay đổi đó cũng được thực hiện trên firebase. Việc thay đổi này là cần thiết vì firebase cho phép chúng ta tạo ra ...
Trong quá trình sử dụng Firebase, sẽ có lúc bạn cần đồng bộ dữ liệu của mình lên firebase mỗi khi có sự thay đổi. Bạn muốn mỗi khi có sự thay đổi ở bảng users trên DB của mình thì sự thay đổi đó cũng được thực hiện trên firebase. Việc thay đổi này là cần thiết vì firebase cho phép chúng ta tạo ra ứng dụng realtime thì thông tin được thay đổi sẽ cập nhật ngay lập tức. Ví dụ: trong ứng dụng chat realtime thì mỗi khi người dùng cập nhật tên, avatar thì ở màn hình chat, bạn cũng phải cập nhật tên, avatar của member đang chat đúng không nào?
Hôm nay, mình xin giới thiệu với các bạn cách đồng bộ hóa các Eloquent model của bạn với Firebase Realtime Database một cách đơn giản.
Với các bạn chưa từng làm việc với Firebase, bạn có thể bắt đầu thử nghiệm với Firebase bằng cách tạo dự án trên Firebase Console. Sau khi bạn đăng nhập bằng tài khoản Google của mình, hãy tạo một dự án mới.
Khi dự án được tạo thành công, hãy nhấn "Add Firebase to your web app" và sẽ xuất hiện cửa sổ hiển thị cho bạn thông tin đăng nhập API mà bạn sẽ sử dụng để truy cập vào Firebase.
Ứng dụng Laravel của bạn sẽ cần phải được xác thực với Firebase để có quyền ghi dữ liệu vào nó, bạn cần copy Database Secrets lấy từ "Project settings / Service accounts / Database Secrets".
Bây giờ bạn đã thu thập tất cả các thông tin cần thiết từ Firebase, chúng ta sẽ tiếp tục với việc tích hợp vào Laravel.
Trong bài viết này, mình sẽ ví dụ việc đồng bộ lên Firebase với bảng users (mặc định trong Laravel) có các thông tin như sau:
Schema::create('users', function (Blueprint $table) { $table->increments('id'); $table->string('name'); $table->string('email')->unique(); $table->string('password'); $table->rememberToken(); $table->timestamps(); });
Cài đặt Firebase Sync bằng lệnh:
composer require mpociot/laravel-firebase-sync
Tiếp theo, trong file config/services.php, chúng ta thêm 1 array mới với các thông tin của firebase.
'firebase' => [ 'api_key' => env('FIREBASE_API_KEY'), 'auth_domain' => env('FIREBASE_AUTH_DOMAIN'), 'database_url' => env('FIREBASE_DATABASE_URL'), 'project_id' => env('FIREBASE_PROJECT_ID'), 'storage_bucket' => env('FIREBASE_STORAGE_BUCKET'), 'messaging_sender_id' => env('FIREBASE_MESSAGING_SENDER'), ]
Các thông tin này mình sẽ lưu lại tại file .env:
FIREBASE_API_KEY=AIzaSyC5UJQnSMK17rvCf-gDCfvvSccNWOyoYoU FIREBASE_AUTH_DOMAIN=sync-project-e8ed9.firebaseapp.com FIREBASE_DATABASE_URL=https://sync-project-e8ed9.firebaseio.com FIREBASE_PROJECT_ID=sync-project-e8ed9 FIREBASE_STORAGE_BUCKET=sync-project-e8ed9.appspot.com FIREBASE_MESSAGING_SENDER=121873760342
Tạo model user trong file app/User.php (nếu chưa có vì mặc định Laravel đã tạo cho mình rồi), sau đó thêm use SyncsWithFirebase trong model vào:
<?php namespace App; use IlluminateNotificationsNotifiable; use IlluminateFoundationAuthUser as Authenticatable; use MpociotFirebaseSyncsWithFirebase; class User extends Authenticatable { use Notifiable, SyncsWithFirebase; /** * The attributes that are mass assignable. * * @var array */ protected $fillable = [ 'name', 'email', 'password', ]; /** * The attributes that should be hidden for arrays. * * @var array */ protected $hidden = [ 'password', 'remember_token', ]; }
Thông qua việc use trait SyncsWithFirebase vào model, việc tạo mới, cập nhật và xóa sẽ được đồng bộ với Firebase Realtime Database.
Thêm dữ liệu
Mình sẽ demo việc thêm 1 user mới và nó sẽ đồng bộ lên firebase.
Route::get('/', function () { AppUser::create([ 'name' => 'Tien Dat Duong', 'email' => 'duong.tien.dat@framgia.com', 'password' => bcrypt('123456'), ]); });
Đây là kết quả sau khi mình thêm dữ liệu, ngoài việc dữ liệu sau khi tạo mới sẽ được thêm vào DB của mình thì nó còn được đồng bộ lên firebase như thế này:
Dữ liệu trên DB thật
Dữ liệu trên Firebase Realtime Database
Sửa dữ liệu
Route::get('/', function () { $user = AppUser::where('email', 'duong.tien.dat@framgia.com')->first(); $user->name = 'Updated'; $user->save(); });
Dữ liệu trên DB thật
Dữ liệu trên Firebase Realtime Database
Xóa dữ liệu
Route::get('/', function () { AppUser::find(3)->delete(); });
Bản ghi đã bị xóa trên DB thật
Bản ghi đã không còn tồn tại trên Firebase Realtime Database
Mình sẽ làm 1 ví dụ cụ thể mà việc đồng bộ hóa từ Laravel lên Firebase áp dụng được vào. Đó là mỗi khi có user được thêm/sửa/xóa thì sẽ hiển thị realtime lên màn hình show bên phía client. Mình viết ứng dụng nhỏ này nó bằng VueJS để các bạn tham khảo.
Cài đặt
Để client nhận được dữ liệu từ Firebase, chúng ta cần cài firebase:
npm install firebase
Thêm route
routes/web.blade.php
Route::get('/test', function () { return view('test'); });
Coding
- Truyền config của firebase từ Laravel sang JS resources/views/test.blade.php:
<!doctype html> <html lang="{{ app()->getLocale() }}"> <head> <meta charset="utf-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="awidth=device-awidth, initial-scale=1"> <title>Laravel</title> <link href="https://fonts.googleapis.com/css?family=Lato:100" rel="stylesheet" type="text/css"> {{ Html::style(asset('css/app.css')) }} </head> <body> <div id="app" class="flex-center position-ref full-height"> <example-component></example-component> </div> <script type="text/javascript"> // Initialize Firebase window.firebaseConfig = { apiKey: "{{ config('services.firebase.api_key') }}", authDomain: "{{ config('services.firebase.auth_domain') }}", databaseURL: "{{ config('services.firebase.database_url') }}", storageBucket: "{{ config('services.firebase.storage_bucket') }}", }; </script> {{ Html::script(asset('js/app.js')) }} </body> </html>
- Template resources/assets/js/app.js
require('./bootstrap'); window.Vue = require('vue'); Vue.component('example-component', require('./components/ExampleComponent.vue')); const app = new Vue({ el: '#app' });
ExampleComponent.vue
<template> <div class="container"> <div class="row justify-content-center"> <div class="col-lg-12"> <h1>Users</h1> <table class="table table-bordered"> <thead> <tr> <th>Email</th> <th>Name</th> </tr> </thead> <tbody> <tr v-for="user in users"> <td> {{ user.name }} </td> <td> {{ user.email }} </td> </tr> </tbody> </table> </div> </div> </div> </template> <script> import firebase from 'firebase'; export default { data () { return { users: [] // mảng users chứa thông tin của các user lấy từ firebase về } }, mounted() { firebase.initializeApp(window.firebaseConfig); // Khởi tạo firebase realtime database. firebase.database().ref('users/').on('value', (snapshot) => { this.users = snapshot.val(); // Mỗi khi dữ liệu của users trên Firebase thay đổi, sẽ cập nhật vào mảng users. }); } } </script>
Sau khi code xong file js, bạn cần chạy npm run dev hoặc npm run watch để js hoạt động.
Kết quả
Kết quả mà mình nhận được sẽ là bảng hiển thị dữ liệu của các users được đồng bộ hóa với Firebase Realtime Database.
Khi bạn lấy dữ liệu từ 1 địa chỉ tham chiếu (reference) từ firebase, nó sẽ luôn giữ kết nối với server của firebase và trả về dữ liệu mới nhất qua snapshot. Mỗi khi có sự thay đổi dữ liệu trên firebase, mình đã gán dữ liệu từ snapshot của firebase vào biến users của VueJS. VueJS luôn tự động cập nhật giá trị của 1 biến ngay lập tức khi biến đó có sự thay đổi. Do đó cho phép chúng ta cập nhật users vào bảng ngay lập tức mỗi khi firebase trả dữ liệu realtime về.
Bạn có thể test bằng cách mở 2 tab trên trình duyệt, 1 tab thao tác việc thêm/sửa/xóa user như mình đã làm ở thao tác bên trên, 1 tab theo dõi sự thay đổi của dữ liệu thông qua bảng.
Các bạn có thể tham khảo toàn bộ source code của mình trên Github tại đây.
Simple enough! Quá đơn giản phải không nào? Việc của bạn chỉ đơn giản là use trait SyncsWithFirebase trong model mà bạn cần đồng bộ là xong. Với việc này, bạn sẽ có dữ liệu ở cả 2 cơ sở dữ liệu - một cơ sở dữ liệu realtime của Firebase và cả dữ liệu trong mySQL của bạn để sử dụng với Laravel Eloquent ORM.
Nhưng cái gì cũng có nhược điểm của nó. Có 1 lưu ý rằng đây không phải là "đồng bộ hóa" thực sự, vì dữ liệu được xử lý trong Firebase sẽ không được đồng bộ hóa trở lại ứng dụng Laravel của bạn (đồng bộ hóa 1 chiều Server -> Firebase).
Nếu bạn muốn đồng bộ hóa chiều còn lại từ Firebase -> Server thì mình nghĩ các bạn cần 1 server NodeJS, chạy background 24/7 và lắng nghe mọi sự thay đổi của Firebase. Firebase đã cung cấp các event để xử lý việc lắng nghe các thay đổi (child_added, child_changed, child_removed, child_moved) và bạn có thể dựa vào đó để đồng bộ dữ liệu ngược lại từ Firebase lên Server.
Hy vọng, bài viết của mình sẽ giúp ích cho các bạn trong các dự án mà các bạn triển khai. Cám ơn các bạn đã dành thời gian đọc bài viết của mình!
https://github.com/mpociot/laravel-firebase-sync
http://marcelpociot.de/blog/2016-06-20-synchronise-laravel-eloquent-models-with-firebase