Tìm hiểu các Quy tắc bảo mật cơ sở dữ liệu theo thời gian thực của Firebase

Quy tắc bảo mật của Cơ sở dữ liệu theo thời gian thực của Firebase xác định người có quyền đọc và ghi vào cơ sở dữ liệu của bạn, cách dữ liệu được cấu trúc và những chỉ mục tồn tại. Các quy tắc này nằm trên máy chủ Firebase và luôn được thực thi tự động. Mọi yêu cầu đọc và ghi sẽ chỉ được hoàn tất nếu quy tắc của bạn cho phép. Theo mặc định, quy tắc của bạn không cho phép bất kỳ ai truy cập vào cơ sở dữ liệu. Điều này là để bảo vệ cơ sở dữ liệu của bạn khỏi hành vi lạm dụng cho đến khi bạn có thời gian tuỳ chỉnh quy tắc hoặc thiết lập quy trình xác thực.

Quy tắc bảo mật của Cơ sở dữ liệu theo thời gian thực có cú pháp giống như JavaScript và có 4 loại:

Các loại quy tắc
.read Mô tả việc người dùng có được phép đọc dữ liệu hay không và khi nào thì được phép.
.write Mô tả việc dữ liệu có được phép ghi hay không và khi nào thì được phép.
.validate Xác định giá trị được định dạng đúng sẽ trông như thế nào, liệu giá trị đó có thuộc tính con hay không và kiểu dữ liệu.
.indexOn Chỉ định một phần tử con để lập chỉ mục nhằm hỗ trợ sắp xếp và truy vấn.

Tổng quan về bảo mật của Realtime Database

Firebase Realtime Database cung cấp một bộ công cụ đầy đủ để quản lý tính bảo mật của ứng dụng. Các công cụ này giúp bạn dễ dàng xác thực người dùng, thực thi quyền của người dùng và xác thực dữ liệu đầu vào.

Các ứng dụng chạy bằng Firebase chạy nhiều mã phía máy khách hơn so với các ứng dụng có nhiều ngăn xếp công nghệ khác. Do đó, cách chúng tôi tiếp cận vấn đề bảo mật có thể hơi khác so với những gì bạn đã quen.

Xác thực

Bước đầu tiên thường thấy trong việc bảo mật ứng dụng là xác định người dùng. Quá trình này được gọi là xác thực. Bạn có thể sử dụng Xác thực Firebase để yêu cầu người dùng đăng nhập vào ứng dụng. Xác thực Firebase bao gồm hỗ trợ tích hợp cho các phương thức xác thực phổ biến như Google và Facebook, cũng như đăng nhập bằng email và mật khẩu, đăng nhập ẩn danh và nhiều phương thức khác.

Danh tính người dùng là một khái niệm quan trọng về bảo mật. Người dùng khác nhau có dữ liệu khác nhau và đôi khi có các khả năng khác nhau. Ví dụ: trong một ứng dụng trò chuyện, mỗi tin nhắn được liên kết với người dùng đã tạo tin nhắn đó. Người dùng cũng có thể xoá tin nhắn của chính mình nhưng không thể xoá tin nhắn do người dùng khác đăng.

Ủy quyền

Xác định người dùng chỉ là một phần của bảo mật. Sau khi biết danh tính của họ, bạn cần có cách kiểm soát quyền truy cập của họ vào dữ liệu trong cơ sở dữ liệu. Quy tắc bảo mật của Cơ sở dữ liệu theo thời gian thực cho phép bạn kiểm soát quyền truy cập cho từng người dùng. Ví dụ: sau đây là một tập hợp các quy tắc bảo mật cho phép bất kỳ ai đọc đường dẫn /foo/, nhưng không ai được ghi vào đường dẫn đó:

{
  "rules": {
    "foo": {
      ".read": true,
      ".write": false
    }
  }
}

Các quy tắc .read.write xếp tầng, vì vậy, tập hợp quy tắc này cấp quyền đọc cho mọi dữ liệu ở đường dẫn /foo/ cũng như mọi đường dẫn sâu hơn như /foo/bar/baz. Xin lưu ý rằng các quy tắc .read.write nông hơn trong cơ sở dữ liệu sẽ ghi đè các quy tắc sâu hơn, vì vậy, quyền đọc đối với /foo/bar/baz vẫn sẽ được cấp trong ví dụ này ngay cả khi một quy tắc ở đường dẫn /foo/bar/baz được đánh giá là sai.

Quy tắc bảo mật của Cơ sở dữ liệu theo thời gian thực bao gồm các biến được tạo sẵn và hàm cho phép bạn tham chiếu đến các đường dẫn khác, dấu thời gian phía máy chủ, thông tin xác thực, và nhiều thông tin khác. Sau đây là ví dụ về một quy tắc cấp quyền ghi cho người dùng đã xác thực vào /users/<uid>/, trong đó <uid> là mã nhận dạng của người dùng thu được thông qua Firebase Authentication.

{
  "rules": {
    "users": {
      "$uid": {
        ".write": "$uid === auth.uid"
      }
    }
  }
}

Xác thực dữ liệu

Firebase Realtime Database không có lược đồ. Điều này giúp bạn dễ dàng thay đổi mọi thứ khi phát triển, nhưng sau khi ứng dụng sẵn sàng phân phối, điều quan trọng là dữ liệu phải nhất quán. Ngôn ngữ quy tắc bao gồm một quy tắc .validate cho phép bạn áp dụng logic xác thực bằng cách sử dụng cùng các biểu thức được dùng cho các quy tắc .read.write. Điểm khác biệt duy nhất là các quy tắc xác thực không xếp tầng, vì vậy, tất cả các quy tắc xác thực có liên quan phải được đánh giá là đúng để cho phép ghi.

Các quy tắc này thực thi rằng dữ liệu được ghi vào /foo/ phải là một chuỗi có ít hơn 100 ký tự:

{
  "rules": {
    "foo": {
      ".validate": "newData.isString() && newData.val().length < 100"
    }
  }
}

Các quy tắc xác thực có quyền truy cập vào tất cả các hàm và biến tích hợp sẵn giống như các quy tắc .read.write. Bạn có thể sử dụng các quy tắc này để tạo các quy tắc xác thực nhận biết được dữ liệu ở nơi khác trong cơ sở dữ liệu, danh tính của người dùng, thời gian của máy chủ và nhiều thông tin khác.

Xác định chỉ mục cơ sở dữ liệu

Firebase Realtime Database cho phép sắp xếp và truy vấn dữ liệu. Đối với kích thước dữ liệu nhỏ, cơ sở dữ liệu hỗ trợ truy vấn đặc biệt, vì vậy, thường không cần chỉ mục trong quá trình phát triển. Tuy nhiên, trước khi ra mắt ứng dụng, bạn cần chỉ định chỉ mục cho mọi truy vấn để đảm bảo các truy vấn đó tiếp tục hoạt động khi ứng dụng của bạn phát triển.

Các chỉ mục được chỉ định bằng quy tắc .indexOn. Sau đây là ví dụ về một khai báo chỉ mục sẽ lập chỉ mục các trường chiều cao và chiều dài cho danh sách khủng long:

{
  "rules": {
    "dinosaurs": {
      ".indexOn": ["height", "length"]
    }
  }
}

Các bước tiếp theo