Programowanie lokalne dzięki Pakietowi emulatorów Firebase

1. Zanim zaczniesz

Narzędzia backendu bezserwerowego, takie jak Cloud Firestore i Cloud Functions, są bardzo łatwe w użyciu, ale mogą być trudne do przetestowania. Pakiet emulatorów lokalnych Firebase umożliwia uruchamianie lokalnych wersji tych usług na komputerze deweloperskim, dzięki czemu możesz szybko i bezpiecznie tworzyć aplikacje.

Wymagania wstępne

  • prosty edytor, np. Visual Studio Code, Atom lub Sublime Text;
  • Node.js w wersji 10.0.0 lub nowszej (aby zainstalować Node.js, użyj nvm, a aby sprawdzić wersję, uruchom node --version)
  • Java 7 lub nowsza (aby zainstalować Javę, postępuj zgodnie z tymi instrukcjami, a aby sprawdzić wersję, uruchom java -version)

Co musisz zrobić

W tym ćwiczeniu z programowania uruchomisz i debugujesz prostą aplikację do zakupów online, która korzysta z kilku usług Firebase:

  • Cloud Firestore:skalowalna globalnie, bezserwerowa baza danych NoSQL z funkcjami czasu rzeczywistego.
  • Cloud Functions: bezserwerowy kod backendu, który jest uruchamiany w odpowiedzi na zdarzenia lub żądania HTTP.
  • Uwierzytelnianie Firebase: zarządzana usługa uwierzytelniania, która integruje się z innymi usługami Firebase.
  • Hosting Firebase: szybki i bezpieczny hosting aplikacji internetowych.

Połączysz aplikację z pakietem emulatorów, aby umożliwić lokalne tworzenie.

2589e2f95b74fa88.png

Dowiesz się też, jak:

  • Jak połączyć aplikację z pakietem Emulator Suite i jak połączone są poszczególne emulatory.
  • Jak działają reguły zabezpieczeń Firebase i jak testować reguły zabezpieczeń Firestore na lokalnym emulatorze.
  • Jak napisać funkcję Firebase aktywowaną przez zdarzenia Firestore i jak napisać testy integracyjne, które działają w pakiecie emulatorów.

2. Skonfiguruj

Pobieranie kodu źródłowego

W tym ćwiczeniu zaczniesz od prawie ukończonej wersji przykładowej aplikacji The Fire Store, więc najpierw musisz sklonować kod źródłowy:

$ git clone https://github.com/firebase/emulators-codelab.git

Następnie przejdź do katalogu ćwiczenia, w którym będziesz pracować przez resztę tego ćwiczenia:

$ cd emulators-codelab/codelab-initial-state

Teraz zainstaluj zależności, aby móc uruchomić kod. Jeśli korzystasz z wolniejszego połączenia internetowego, może to potrwać minutę lub dwie:

# Move into the functions directory
$ cd functions

# Install dependencies
$ npm install

# Move back into the previous directory
$ cd ../

Pobieranie wiersza poleceń Firebase

Pakiet emulatorów jest częścią wiersza poleceń Firebase, który możesz zainstalować na swoim komputerze za pomocą tego polecenia:

$ npm install -g firebase-tools

Następnie sprawdź, czy masz najnowszą wersję interfejsu CLI. Ten moduł powinien działać w wersji 9.0.0 lub nowszej, ale nowsze wersje zawierają więcej poprawek błędów.

$ firebase --version
9.6.0

Łączenie z projektem Firebase

Tworzenie projektu Firebase

  1. Zaloguj się w konsoli Firebase, korzystając ze swojego konta Google.
  2. Kliknij przycisk, aby utworzyć nowy projekt, a potem wpisz jego nazwę (np. Emulators Codelab).
  3. Kliknij Dalej.
  4. Po wyświetleniu monitu przeczytaj i zaakceptuj warunki usługi Firebase, a potem kliknij Dalej.
  5. (Opcjonalnie) Włącz w konsoli Firebase pomoc AI (nazywaną „Gemini w Firebase”).
  6. W tym samouczku nie potrzebujesz Google Analytics, więc wyłącz opcję Google Analytics.
  7. Kliknij Utwórz projekt, poczekaj, aż projekt zostanie udostępniony, a następnie kliknij Dalej.

Łączenie kodu z projektem Firebase

Teraz musimy połączyć ten kod z projektem Firebase. Najpierw uruchom to polecenie, aby zalogować się w wierszu poleceń Firebase:

$ firebase login

Następnie uruchom to polecenie, aby utworzyć alias projektu. Zastąp $YOUR_PROJECT_ID identyfikatorem projektu Firebase.

$ firebase use $YOUR_PROJECT_ID

Możesz teraz uruchomić aplikację.

3. Uruchamianie emulatorów

W tej sekcji uruchomisz aplikację lokalnie. Oznacza to, że nadszedł czas na uruchomienie pakietu emulatorów.

Uruchamianie emulatorów

W katalogu źródłowym codelabu uruchom to polecenie, aby uruchomić emulatory:

$ firebase emulators:start --import=./seed

Powinny się wyświetlić dane wyjściowe podobne do tych:

$ firebase emulators:start --import=./seed
i  emulators: Starting emulators: auth, functions, firestore, hosting
⚠  functions: The following emulators are not running, calls to these services from the Functions emulator will affect production: database, pubsub
i  firestore: Importing data from /Users/samstern/Projects/emulators-codelab/codelab-initial-state/seed/firestore_export/firestore_export.overall_export_metadata
i  firestore: Firestore Emulator logging to firestore-debug.log
i  hosting: Serving hosting files from: public
✔  hosting: Local server: http://127.0.0.1:5000
i  ui: Emulator UI logging to ui-debug.log
i  functions: Watching "/Users/samstern/Projects/emulators-codelab/codelab-initial-state/functions" for Cloud Functions...
✔  functions[calculateCart]: firestore function initialized.

┌─────────────────────────────────────────────────────────────┐
│ ✔  All emulators ready! It is now safe to connect your app. │
│ i  View Emulator UI at http://127.0.0.1:4000                │
└─────────────────────────────────────────────────────────────┘

┌────────────────┬────────────────┬─────────────────────────────────┐
│ Emulator       │ Host:Port      │ View in Emulator UI             │
├────────────────┼────────────────┼─────────────────────────────────┤
│ Authentication │ 127.0.0.1:9099 │ http://127.0.0.1:4000/auth      │
├────────────────┼────────────────┼─────────────────────────────────┤
│ Functions      │ 127.0.0.1:5001 │ http://127.0.0.1:4000/functions │
├────────────────┼────────────────┼─────────────────────────────────┤
│ Firestore      │ 127.0.0.1:8080 │ http://127.0.0.1:4000/firestore │
├────────────────┼────────────────┼─────────────────────────────────┤
│ Hosting        │ 127.0.0.1:5000 │ n/a                             │
└────────────────┴────────────────┴─────────────────────────────────┘
  Emulator Hub running at 127.0.0.1:4400
  Other reserved ports: 4500

Issues? Report them at https://github.com/firebase/firebase-tools/issues and attach the *-debug.log files.

Gdy zobaczysz komunikat All emulators started (Wszystkie emulatory zostały uruchomione), aplikacja będzie gotowa do użycia.

Łączenie aplikacji internetowej z emulatorami

Z tabeli w dziennikach wynika, że emulator Cloud Firestore nasłuchuje na porcie 8080, a emulator Authentication – na porcie 9099.

┌────────────────┬────────────────┬─────────────────────────────────┐
│ Emulator       │ Host:Port      │ View in Emulator UI             │
├────────────────┼────────────────┼─────────────────────────────────┤
│ Authentication │ 127.0.0.1:9099 │ http://127.0.0.1:4000/auth      │
├────────────────┼────────────────┼─────────────────────────────────┤
│ Functions      │ 127.0.0.1:5001 │ http://127.0.0.1:4000/functions │
├────────────────┼────────────────┼─────────────────────────────────┤
│ Firestore      │ 127.0.0.1:8080 │ http://127.0.0.1:4000/firestore │
├────────────────┼────────────────┼─────────────────────────────────┤
│ Hosting        │ 127.0.0.1:5000 │ n/a                             │
└────────────────┴────────────────┴─────────────────────────────────┘

Połączmy kod interfejsu z emulatorem, a nie z wersją produkcyjną. Otwórz plik public/js/homepage.js i znajdź funkcję onDocumentReady. Widzimy, że kod uzyskuje dostęp do standardowych instancji Firestore i Auth:

public/js/homepage.js

  const auth = firebaseApp.auth();
  const db = firebaseApp.firestore();

Zaktualizujmy obiekty dbauth, aby wskazywały lokalne emulatory:

public/js/homepage.js

  const auth = firebaseApp.auth();
  const db = firebaseApp.firestore();

  // ADD THESE LINES
  if (location.hostname === "127.0.0.1") {
    console.log("127.0.0.1 detected!");
    auth.useEmulator("http://127.0.0.1:9099");
    db.useEmulator("127.0.0.1", 8080);
  }

Teraz, gdy aplikacja działa na Twoim komputerze (obsługiwana przez emulator hostingu), klient Firestore również wskazuje lokalny emulator, a nie produkcyjną bazę danych.

Otwórz interfejs EmulatorUI

W przeglądarce otwórz stronę http://127.0.0.1:4000/. Powinien pojawić się interfejs Pakietu emulatorów.

Ekran główny interfejsu emulatorów

Kliknij, aby zobaczyć interfejs emulatora Firestore. Kolekcja items zawiera już dane zaimportowane za pomocą flagi --import.

4ef88d0148405d36.png

4. Uruchamianie aplikacji

Włącz aplikację

W przeglądarce otwórz adres http://127.0.0.1:5000. Powinna się wyświetlić aplikacja The Fire Store działająca lokalnie na Twoim komputerze.

939f87946bac2ee4.png

Korzystanie z aplikacji

Wybierz produkt na stronie głównej i kliknij Dodaj do koszyka. Niestety pojawi się ten błąd:

a11bd59933a8e885.png

Naprawmy ten błąd. Wszystko działa w emulatorach, więc możemy eksperymentować bez obawy o wpływ na rzeczywiste dane.

5. Debugowanie aplikacji

Znajdź błąd

Spójrzmy na konsolę deweloperską Chrome. Naciśnij Control+Shift+J (Windows, Linux, ChromeOS) lub Command+Option+J (Mac), aby wyświetlić błąd w konsoli:

74c45df55291dab1.png

Wygląda na to, że w metodzie addToCart wystąpił błąd. Przyjrzyjmy się temu. Gdzie w tej metodzie próbujemy uzyskać dostęp do czegoś, co nazywa się uid, i dlaczego może to być null? Obecnie metoda wygląda tak w public/js/homepage.js:

public/js/homepage.js

  addToCart(id, itemData) {
    console.log("addToCart", id, JSON.stringify(itemData));
    return this.db
      .collection("carts")
      .doc(this.auth.currentUser.uid)
      .collection("items")
      .doc(id)
      .set(itemData);
  }

Aha! Nie jesteśmy zalogowani w aplikacji. Zgodnie z dokumentacją Uwierzytelniania Firebase, gdy nie jesteśmy zalogowani, auth.currentUser ma wartość null. Dodajmy sprawdzenie:

public/js/homepage.js

  addToCart(id, itemData) {
    // ADD THESE LINES
    if (this.auth.currentUser === null) {
      this.showError("You must be signed in!");
      return;
    }

    // ...
  }

Testowanie aplikacji

Teraz odśwież stronę, a następnie kliknij Dodaj do koszyka. Tym razem powinien pojawić się bardziej czytelny błąd:

c65f6c05588133f7.png

Jeśli jednak na pasku narzędzi u góry klikniesz Zaloguj się, a potem ponownie Dodaj do koszyka, zobaczysz, że koszyk został zaktualizowany.

Wygląda jednak na to, że liczby są nieprawidłowe:

239f26f02f959eef.png

Nie martw się, wkrótce naprawimy ten błąd. Najpierw przyjrzyjmy się dokładnie temu, co się dzieje, gdy dodajesz produkt do koszyka.

6. Aktywatory funkcji lokalnych

Kliknięcie Dodaj do koszyka powoduje uruchomienie łańcucha zdarzeń, w których uczestniczy wiele emulatorów. W logach wiersza poleceń Firebase po dodaniu produktu do koszyka powinny pojawić się komunikaty podobne do tych:

i  functions: Beginning execution of "calculateCart"
i  functions: Finished "calculateCart" in ~1s

Aby wygenerować te logi i zaktualizować interfejs, wystąpiły 4 kluczowe zdarzenia:

68c9323f2ad10f7a.png

1) Zapis w Firestore – klient

Do kolekcji Firestore /carts/{cartId}/items/{itemId}/ dodawany jest nowy dokument. Ten kod możesz zobaczyć w funkcji addToCart w sekcji public/js/homepage.js:

public/js/homepage.js

  addToCart(id, itemData) {
    // ...
    console.log("addToCart", id, JSON.stringify(itemData));
    return this.db
      .collection("carts")
      .doc(this.auth.currentUser.uid)
      .collection("items")
      .doc(id)
      .set(itemData);
  }

2. Wywołanie funkcji w Cloud Functions

Funkcja w Cloud Functions calculateCart nasłuchuje zdarzeń zapisu (tworzenia, aktualizowania lub usuwania) dotyczących produktów w koszyku za pomocą wyzwalacza onWrite, który możesz zobaczyć w functions/index.js:

functions/index.js

exports.calculateCart = functions.firestore
    .document("carts/{cartId}/items/{itemId}")
    .onWrite(async (change, context) => {
      try {
        let totalPrice = 125.98;
        let itemCount = 8;

        const cartRef = db.collection("carts").doc(context.params.cartId);

        await cartRef.update({
          totalPrice,
          itemCount
        });
      } catch(err) {
      }
    }
);

3. Zapis w Firestore – administrator

Funkcja calculateCart odczytuje wszystkie produkty w koszyku i sumuje łączną liczbę oraz cenę, a następnie aktualizuje dokument „cart” o nowe sumy (patrz cartRef.update(...) powyżej).

4) Odczyt Firestore – klient

Interfejs internetowy jest subskrybowany, aby otrzymywać aktualizacje dotyczące zmian w koszyku. Po zapisaniu nowych sum przez funkcję w Cloud Functions i zaktualizowaniu interfejsu użytkownika otrzymuje ona aktualizację w czasie rzeczywistym, co widać na ilustracji public/js/homepage.js:

public/js/homepage.js

this.cartUnsub = cartRef.onSnapshot(cart => {
   // The cart document was changed, update the UI
   // ...
});

Podsumowanie

Dobra robota! Właśnie skonfigurowano w pełni lokalną aplikację, która korzysta z 3 różnych emulatorów Firebase do testowania w pełni lokalnego.

db82eef1706c9058.gif

Poczekaj, to jeszcze nie wszystko. W następnej sekcji dowiesz się:

  • Jak pisać testy jednostkowe, które korzystają z emulatorów Firebase.
  • Jak używać emulatorów Firebase do debugowania reguł zabezpieczeń.

7. Tworzenie reguł zabezpieczeń dostosowanych do aplikacji

Nasza aplikacja internetowa odczytuje i zapisuje dane, ale do tej pory nie martwiliśmy się zbytnio o bezpieczeństwo. Cloud Firestore używa systemu o nazwie „Reguły zabezpieczeń”, aby określić, kto ma dostęp do odczytu i zapisu danych. Pakiet emulatorów to świetny sposób na prototypowanie tych reguł.

W edytorze otwórz plik emulators-codelab/codelab-initial-state/firestore.rules. Jak widzisz, nasze zasady dzielą się na 3 główne sekcje:

rules_version = '2';
service cloud.firestore {
  match /databases/{database}/documents {
    // User's cart metadata
    match /carts/{cartID} {
      // TODO: Change these! Anyone can read or write.
      allow read, write: if true;
    }

    // Items inside the user's cart
    match /carts/{cartID}/items/{itemID} {
      // TODO: Change these! Anyone can read or write.
      allow read, write: if true;
    }

    // All items available in the store. Users can read
    // items but never write them.
    match /items/{itemID} {
      allow read: if true;
    }
  }
}

Obecnie każdy może odczytywać i zapisywać dane w naszej bazie danych. Chcemy mieć pewność, że tylko prawidłowe operacje są przeprowadzane i że nie wyciekają żadne informacje poufne.

W tym laboratorium, zgodnie z zasadą najmniejszych uprawnień, zablokujemy wszystkie dokumenty i stopniowo będziemy dodawać dostęp, aż wszyscy użytkownicy uzyskają potrzebne uprawnienia, ale nie więcej. Zaktualizujmy pierwsze 2 reguły, aby odmawiać dostępu, ustawiając warunek na false:

rules_version = '2';
service cloud.firestore {
  match /databases/{database}/documents {
    // User's cart metadata
    match /carts/{cartID} {
      // UPDATE THIS LINE
      allow read, write: if false;
    }

    // Items inside the user's cart
    match /carts/{cartID}/items/{itemID} {
      // UPDATE THIS LINE
      allow read, write: if false;
    }

    // All items available in the store. Users can read
    // items but never write them.
    match /items/{itemID} {
      allow read: if true;
    }
  }
}

8. Uruchamianie emulatorów i testów

Uruchom emulatory

W wierszu poleceń sprawdź, czy jesteś w katalogu emulators-codelab/codelab-initial-state/. Emulatory mogą być nadal uruchomione z poprzednich kroków. Jeśli nie, uruchom emulatory ponownie:

$ firebase emulators:start --import=./seed

Gdy emulatory będą działać, możesz przeprowadzać na nich testy lokalne.

Przeprowadzanie testów

W wierszu poleceń w nowej karcie terminala w katalogu emulators-codelab/codelab-initial-state/

Najpierw przejdź do katalogu funkcji (pozostaniemy w nim do końca tego laboratorium):

$ cd functions

Teraz uruchom testy Mocha w katalogu funkcji i przewiń do góry:

# Run the tests
$ npm test

> functions@ test .../emulators-codelab/codelab-initial-state/functions
> mocha

  shopping carts
    1) can be created and updated by the cart owner
    2) can be read only by the cart owner

  shopping cart items
    3) can be read only by the cart owner
    4) can be added only by the cart owner

  adding an item to the cart recalculates the cart total. 
    - should sum the cost of their items


  0 passing (364ms)
  1 pending
  4 failing

Obecnie mamy 4 błędy. W miarę tworzenia pliku reguł możesz mierzyć postępy, obserwując, jak przechodzą kolejne testy.

9. Bezpieczny dostęp do koszyka

Pierwsze 2 błędy to testy „koszyka”, które sprawdzają, czy:

  • Użytkownicy mogą tworzyć i aktualizować tylko własne koszyki.
  • Użytkownicy mogą odczytywać tylko własne koszyki.

functions/test.js

  it('can be created and updated by the cart owner', async () => {
    // Alice can create her own cart
    await firebase.assertSucceeds(aliceDb.doc("carts/alicesCart").set({
      ownerUID: "alice",
      total: 0
    }));

    // Bob can't create Alice's cart
    await firebase.assertFails(bobDb.doc("carts/alicesCart").set({
      ownerUID: "alice",
      total: 0
    }));

    // Alice can update her own cart with a new total
    await firebase.assertSucceeds(aliceDb.doc("carts/alicesCart").update({
      total: 1
    }));

    // Bob can't update Alice's cart with a new total
    await firebase.assertFails(bobDb.doc("carts/alicesCart").update({
      total: 1
    }));
  });

  it("can be read only by the cart owner", async () => {
    // Setup: Create Alice's cart as admin
    await admin.doc("carts/alicesCart").set({
      ownerUID: "alice",
      total: 0
    });

    // Alice can read her own cart
    await firebase.assertSucceeds(aliceDb.doc("carts/alicesCart").get());

    // Bob can't read Alice's cart
    await firebase.assertFails(bobDb.doc("carts/alicesCart").get());
  });

Sprawmy, aby te testy zakończyły się powodzeniem. W edytorze otwórz plik reguł zabezpieczeń firestore.rules i zaktualizuj instrukcje w match /carts/{cartID}:

firestore.rules

rules_version = '2';
service cloud.firestore {
    // UPDATE THESE LINES
    match /carts/{cartID} {
      allow create: if request.auth.uid == request.resource.data.ownerUID;
      allow read, update, delete: if request.auth.uid == resource.data.ownerUID;
    }

    // ...
  }
}

Te reguły zezwalają teraz tylko właścicielowi koszyka na dostęp do odczytu i zapisu.

Do weryfikacji przychodzących danych i uwierzytelniania użytkownika używamy 2 obiektów dostępnych w kontekście każdej reguły:

10. Testowanie dostępu do koszyka

Pakiet Emulator Suite automatycznie aktualizuje reguły po zapisaniu firestore.rules. Aby sprawdzić, czy emulator ma zaktualizowane reguły, poszukaj na karcie z uruchomionym emulatorem komunikatu Rules updated:

5680da418b420226.png

Ponownie uruchom testy i sprawdź, czy 2 pierwsze testy zakończyły się powodzeniem:

$ npm test

> functions@ test .../emulators-codelab/codelab-initial-state/functions
> mocha

  shopping carts
    ✓ can be created and updated by the cart owner (195ms)
    ✓ can be read only by the cart owner (136ms)

  shopping cart items
    1) can be read only by the cart owner
    2) can be added only by the cart owner

  adding an item to the cart recalculates the cart total. 
    - should sum the cost of their items

  2 passing (482ms)
  1 pending
  2 failing

Dobra robota! Masz teraz dostęp do koszyków. Przejdźmy do następnego testu, który się nie powiódł.

11. Sprawdź proces „Dodaj do koszyka” w interfejsie

Obecnie właściciele koszyków mogą odczytywać i zapisywać dane w koszyku, ale nie mogą odczytywać ani zapisywać poszczególnych produktów w koszyku. Dzieje się tak, ponieważ właściciele mają dostęp do dokumentu koszyka, ale nie mają dostępu do podzbioru elementów koszyka.

Jest to stan nieprawidłowy dla użytkowników.

Wróć do interfejsu internetowego działającego na http://127.0.0.1:5000, i spróbuj dodać coś do koszyka. W konsoli debugowania zobaczysz błąd Permission Denied, ponieważ nie przyznaliśmy jeszcze użytkownikom dostępu do utworzonych dokumentów w podzbiorze items.

12. Zezwalaj na dostęp do produktów w koszyku

Te 2 testy potwierdzają, że użytkownicy mogą dodawać produkty tylko do własnego koszyka i tylko z niego je odczytywać:

  it("can be read only by the cart owner", async () => {
    // Alice can read items in her own cart
    await firebase.assertSucceeds(aliceDb.doc("carts/alicesCart/items/milk").get());

    // Bob can't read items in alice's cart
    await firebase.assertFails(bobDb.doc("carts/alicesCart/items/milk").get())
  });

  it("can be added only by the cart owner",  async () => {
    // Alice can add an item to her own cart
    await firebase.assertSucceeds(aliceDb.doc("carts/alicesCart/items/lemon").set({
      name: "lemon",
      price: 0.99
    }));

    // Bob can't add an item to alice's cart
    await firebase.assertFails(bobDb.doc("carts/alicesCart/items/lemon").set({
      name: "lemon",
      price: 0.99
    }));
  });

Możemy więc napisać regułę, która zezwala na dostęp, jeśli bieżący użytkownik ma ten sam identyfikator UID co właściciel UID w dokumencie koszyka. Nie musisz określać różnych reguł dla create, update, delete, więc możesz użyć reguły write, która ma zastosowanie do wszystkich żądań modyfikujących dane.

Zaktualizuj regułę dla dokumentów w podkolekcji elementów. Symbol get w warunku odczytuje wartość z Firestore – w tym przypadku ownerUID w dokumencie koszyka.

rules_version = '2';
service cloud.firestore {
  match /databases/{database}/documents {
    // ...

    // UPDATE THESE LINES
    match /carts/{cartID}/items/{itemID} {
      allow read, write: if get(/databases/$(database)/documents/carts/$(cartID)).data.ownerUID == request.auth.uid;
    }

    // ...
  }
}

13. Testowanie dostępu do produktów w koszyku

Teraz możemy ponownie uruchomić test. Przewiń do góry i sprawdź, czy więcej testów zostało zaliczonych:

$ npm test

> functions@ test .../emulators-codelab/codelab-initial-state/functions
> mocha

  shopping carts
    ✓ can be created and updated by the cart owner (195ms)
    ✓ can be read only by the cart owner (136ms)

  shopping cart items
    ✓ can be read only by the cart owner (111ms)
    ✓ can be added only by the cart owner


  adding an item to the cart recalculates the cart total. 
    - should sum the cost of their items


  4 passing (401ms)
  1 pending

Świetnie! Teraz wszystkie nasze testy są zaliczone. Mamy 1 test w toku, ale do tego wrócimy za kilka kroków.

14. Ponownie sprawdź proces „dodawania do koszyka”

Wróć do interfejsu internetowego ( http://127.0.0.1:5000) i dodaj produkt do koszyka. To ważny krok, który pozwala potwierdzić, że nasze testy i reguły są zgodne z funkcjami wymaganymi przez klienta. (Pamiętaj, że ostatnim razem, gdy testowaliśmy interfejs, użytkownicy nie mogli dodawać produktów do koszyka).

69ad26cee520bf24.png

Gdy firestore.rules zostanie zapisany, klient automatycznie ponownie wczyta reguły. Spróbuj więc dodać coś do koszyka.

Podsumowanie

Dobra robota! Właśnie zwiększono bezpieczeństwo aplikacji, co jest niezbędnym krokiem do przygotowania jej do wdrożenia w wersji produkcyjnej. Gdyby to była aplikacja produkcyjna, moglibyśmy dodać te testy do naszego potoku ciągłej integracji. Dzięki temu będziemy mieć pewność, że dane koszyka będą objęte kontrolą dostępu, nawet jeśli inni użytkownicy będą modyfikować reguły.

ba5440b193e75967.gif

Ale to nie wszystko!

Jeśli przejdziesz dalej, dowiesz się:

  • Jak napisać funkcję aktywowaną przez zdarzenie Firestore
  • Tworzenie testów działających na wielu emulatorach

15. Konfigurowanie testów Cloud Functions

Do tej pory skupialiśmy się na interfejsie aplikacji internetowej i regułach zabezpieczeń Firestore. Ta aplikacja korzysta też z Cloud Functions, aby aktualizować koszyk użytkownika, więc chcemy też przetestować ten kod.

Pakiet emulatorów ułatwia testowanie Cloud Functions, nawet tych, które korzystają z Cloud Firestore i innych usług.

W edytorze otwórz plik emulators-codelab/codelab-initial-state/functions/test.js i przewiń go do ostatniego testu. Obecnie jest on oznaczony jako oczekujący:

//  REMOVE .skip FROM THIS LINE
describe.skip("adding an item to the cart recalculates the cart total. ", () => {
  // ...

  it("should sum the cost of their items", async () => {
    ...
  });
});

Aby włączyć test, usuń znak .skip, tak aby wyglądało to tak:

describe("adding an item to the cart recalculates the cart total. ", () => {
  // ...

  it("should sum the cost of their items", async () => {
    ...
  });
});

Następnie znajdź zmienną REAL_FIREBASE_PROJECT_ID na początku pliku i zmień ją na prawdziwy identyfikator projektu Firebase:

// CHANGE THIS LINE
const REAL_FIREBASE_PROJECT_ID = "changeme";

Jeśli zapomnisz identyfikatora projektu, możesz go znaleźć w ustawieniach projektu w konsoli Firebase:

d6d0429b700d2b21.png

16. Przeglądanie testów funkcji

Ten test weryfikuje interakcję między Cloud Firestore a Cloud Functions, dlatego wymaga więcej konfiguracji niż testy w poprzednich ćwiczeniach. Przyjrzyjmy się temu testowi i zobaczmy, czego się od Ciebie oczekuje.

Tworzenie koszyka

Funkcje Cloud działają w zaufanym środowisku serwera i mogą korzystać z uwierzytelniania konta usługi używanego przez pakiet Admin SDK . Najpierw inicjujesz aplikację za pomocą funkcji initializeAdminApp zamiast initializeApp. Następnie tworzysz DocumentReference dla koszyka, do którego będziemy dodawać produkty, i inicjujesz koszyk:

it("should sum the cost of their items", async () => {
    const db = firebase
        .initializeAdminApp({ projectId: REAL_FIREBASE_PROJECT_ID })
        .firestore();

    // Setup: Initialize cart
    const aliceCartRef = db.doc("carts/alice")
    await aliceCartRef.set({ ownerUID: "alice", totalPrice: 0 });

    ...
  });

Aktywowanie funkcji

Następnie dodaj dokumenty do podkolekcji items dokumentu koszyka, aby wywołać funkcję. Dodaj 2 elementy, aby sprawdzić, czy funkcja dodaje je prawidłowo.

it("should sum the cost of their items", async () => {
    const db = firebase
        .initializeAdminApp({ projectId: REAL_FIREBASE_PROJECT_ID })
        .firestore();

    // Setup: Initialize cart
    const aliceCartRef = db.doc("carts/alice")
    await aliceCartRef.set({ ownerUID: "alice", totalPrice: 0 });

    //  Trigger calculateCart by adding items to the cart
    const aliceItemsRef = aliceCartRef.collection("items");
    await aliceItemsRef.doc("doc1").set({name: "nectarine", price: 2.99});
    await aliceItemsRef.doc("doc2").set({ name: "grapefruit", price: 6.99 });

    ...
    });
  });

Określ oczekiwania dotyczące testu

Użyj onSnapshot(), aby zarejestrować odbiorcę wszelkich zmian w dokumencie koszyka. onSnapshot() zwraca funkcję, której możesz użyć do wyrejestrowania detektora.

W tym teście dodaj 2 produkty, które razem kosztują 9,98 PLN. Następnie sprawdź, czy koszyk zawiera oczekiwane wartości itemCounttotalPrice. Jeśli tak, funkcja spełniła swoje zadanie.

it("should sum the cost of their items", (done) => {
    const db = firebase
        .initializeAdminApp({ projectId: REAL_FIREBASE_PROJECT_ID })
        .firestore();

    // Setup: Initialize cart
    const aliceCartRef = db.doc("carts/alice")
    aliceCartRef.set({ ownerUID: "alice", totalPrice: 0 });

    //  Trigger calculateCart by adding items to the cart
    const aliceItemsRef = aliceCartRef.collection("items");
    aliceItemsRef.doc("doc1").set({name: "nectarine", price: 2.99});
    aliceItemsRef.doc("doc2").set({ name: "grapefruit", price: 6.99 });
    
    // Listen for every update to the cart. Every time an item is added to
    // the cart's subcollection of items, the function updates `totalPrice`
    // and `itemCount` attributes on the cart.
    // Returns a function that can be called to unsubscribe the listener.
    await new Promise((resolve) => {
      const unsubscribe = aliceCartRef.onSnapshot(snap => {
        // If the function worked, these will be cart's final attributes.
        const expectedCount = 2;
        const expectedTotal = 9.98;
  
        // When the `itemCount`and `totalPrice` match the expectations for the
        // two items added, the promise resolves, and the test passes.
        if (snap.data().itemCount === expectedCount && snap.data().totalPrice == expectedTotal) {
          // Call the function returned by `onSnapshot` to unsubscribe from updates
          unsubscribe();
          resolve();
        };
      });
    });
   });
 });

17. Przeprowadzanie testów

Emulatory z poprzednich testów mogą być nadal uruchomione. Jeśli nie, uruchom emulatory. W wierszu poleceń wpisz

$ firebase emulators:start --import=./seed

Otwórz nową kartę terminala (pozostaw emulatory uruchomione) i przejdź do katalogu funkcji. Może być ona nadal otwarta po testach reguł zabezpieczeń.

$ cd functions

Uruchom teraz testy jednostkowe. Powinno się pojawić 5 testów:

$ npm test

> functions@ test .../emulators-codelab/codelab-initial-state/functions
> mocha

  shopping cart creation
    ✓ can be created by the cart owner (82ms)

  shopping cart reads, updates, and deletes
    ✓ cart can be read by the cart owner (42ms)

  shopping cart items
    ✓ items can be read by the cart owner (40ms)
    ✓ items can be added by the cart owner

  adding an item to the cart recalculates the cart total. 
    1) should sum the cost of their items

  4 passing (2s)
  1 failing

Jeśli przyjrzysz się konkretnemu błędowi, okaże się, że jest to błąd przekroczenia limitu czasu. Dzieje się tak, ponieważ test czeka na prawidłową aktualizację funkcji, ale nigdy się ona nie pojawia. Teraz możemy napisać funkcję, która spełni wymagania testu.

18. Napisz funkcję

Aby naprawić ten test, musisz zaktualizować funkcję w functions/index.js. Chociaż część tej funkcji jest napisana, nie jest ona kompletna. Obecnie funkcja wygląda tak:

// Recalculates the total cost of a cart; triggered when there's a change
// to any items in a cart.
exports.calculateCart = functions
    .firestore.document("carts/{cartId}/items/{itemId}")
    .onWrite(async (change, context) => {
      console.log(`onWrite: ${change.after.ref.path}`);
      if (!change.after.exists) {
        // Ignore deletes
        return;
      }

      let totalPrice = 125.98;
      let itemCount = 8;
      try {
        
        const cartRef = db.collection("carts").doc(context.params.cartId);

        await cartRef.update({
          totalPrice,
          itemCount
        });
      } catch(err) {
      }
    });

Funkcja prawidłowo ustawia odniesienie do koszyka, ale zamiast obliczać wartości totalPriceitemCount, aktualizuje je do wartości zakodowanych na stałe.

Pobierz i przejdź przez

items subcollection

Zainicjuj nową stałą itemsSnap jako podzbiór items. Następnie przeiteruj wszystkie dokumenty w kolekcji.

// Recalculates the total cost of a cart; triggered when there's a change
// to any items in a cart.
exports.calculateCart = functions
    .firestore.document("carts/{cartId}/items/{itemId}")
    .onWrite(async (change, context) => {
      console.log(`onWrite: ${change.after.ref.path}`);
      if (!change.after.exists) {
        // Ignore deletes
        return;
      }


      try {
        let totalPrice = 125.98;
        let itemCount = 8;

        const cartRef = db.collection("carts").doc(context.params.cartId);
        // ADD LINES FROM HERE
        const itemsSnap = await cartRef.collection("items").get();

        itemsSnap.docs.forEach(item => {
          const itemData = item.data();
        })
        // TO HERE
       
        return cartRef.update({
          totalPrice,
          itemCount
        });
      } catch(err) {
      }
    });

Obliczanie wartości totalPrice i itemCount

Najpierw zainicjujmy wartości totalPriceitemCount jako zero.

Następnie dodaj logikę do bloku iteracji. Najpierw sprawdź, czy produkt ma cenę. Jeśli element nie ma określonej ilości, ustaw domyślną wartość 1. Następnie dodaj tę liczbę do bieżącej sumy itemCount. Na koniec dodaj cenę produktu pomnożoną przez ilość do bieżącej sumy totalPrice:

// Recalculates the total cost of a cart; triggered when there's a change
// to any items in a cart.
exports.calculateCart = functions
    .firestore.document("carts/{cartId}/items/{itemId}")
    .onWrite(async (change, context) => {
      console.log(`onWrite: ${change.after.ref.path}`);
      if (!change.after.exists) {
        // Ignore deletes
        return;
      }

      try {
        // CHANGE THESE LINES
        let totalPrice = 0;
        let itemCount = 0;

        const cartRef = db.collection("carts").doc(context.params.cartId);
        const itemsSnap = await cartRef.collection("items").get();

        itemsSnap.docs.forEach(item => {
          const itemData = item.data();
          // ADD LINES FROM HERE
          if (itemData.price) {
            // If not specified, the quantity is 1
            const quantity = itemData.quantity ? itemData.quantity : 1;
            itemCount += quantity;
            totalPrice += (itemData.price * quantity);
          }
          // TO HERE
        })

        await cartRef.update({
          totalPrice,
          itemCount
        });
      } catch(err) {
      }
    });

Możesz też dodać logowanie, aby ułatwić debugowanie stanów powodzenia i błędu:

// Recalculates the total cost of a cart; triggered when there's a change
// to any items in a cart.
exports.calculateCart = functions
    .firestore.document("carts/{cartId}/items/{itemId}")
    .onWrite(async (change, context) => {
      console.log(`onWrite: ${change.after.ref.path}`);
      if (!change.after.exists) {
        // Ignore deletes
        return;
      }

      let totalPrice = 0;
      let itemCount = 0;
      try {
        const cartRef = db.collection("carts").doc(context.params.cartId);
        const itemsSnap = await cartRef.collection("items").get();

        itemsSnap.docs.forEach(item => {
          const itemData = item.data();
          if (itemData.price) {
            // If not specified, the quantity is 1
            const quantity = (itemData.quantity) ? itemData.quantity : 1;
            itemCount += quantity;
            totalPrice += (itemData.price * quantity);
          }
        });

        await cartRef.update({
          totalPrice,
          itemCount
        });

        // OPTIONAL LOGGING HERE
        console.log("Cart total successfully recalculated: ", totalPrice);
      } catch(err) {
        // OPTIONAL LOGGING HERE
        console.warn("update error", err);
      }
    });

19. Ponowne uruchamianie testów

W wierszu poleceń sprawdź, czy emulatory nadal działają, i ponownie uruchom testy. Nie musisz ponownie uruchamiać emulatorów, ponieważ automatycznie wykrywają one zmiany w funkcjach. Wszystkie testy powinny zakończyć się powodzeniem:

$ npm test
> functions@ test .../emulators-codelab/codelab-initial-state/functions
> mocha

  shopping cart creation
    ✓ can be created by the cart owner (306ms)

  shopping cart reads, updates, and deletes
    ✓ cart can be read by the cart owner (59ms)

  shopping cart items
    ✓ items can be read by the cart owner
    ✓ items can be added by the cart owner

  adding an item to the cart recalculates the cart total. 
    ✓ should sum the cost of their items (800ms)


  5 passing (1s)

Dobra robota!

20. Wypróbuj interfejs witryny sklepowej

Aby przeprowadzić ostatni test, wróć do aplikacji internetowej ( http://127.0.0.1:5000/) i dodaj produkt do koszyka.

69ad26cee520bf24.png

Sprawdź, czy koszyk został zaktualizowany i wyświetla prawidłową łączną kwotę. Fantastycznie!

Podsumowanie

Przetestowaliśmy złożony przypadek użycia Cloud Functions dla Firebase i Cloud Firestore. Aby test zakończył się powodzeniem, musisz napisać funkcję w Cloud Functions. Sprawdzisz też, czy nowa funkcja działa w interfejsie. Wszystkie te czynności zostały wykonane lokalnie, a emulatory były uruchamiane na Twoim komputerze.

Utworzono też klienta internetowego, który działa na lokalnych emulatorach, dostosowano reguły zabezpieczeń w celu ochrony danych i przetestowano je za pomocą lokalnych emulatorów.

c6a7aeb91fe97a64.gif