ऐप्लिकेशन बनाते समय, हो सकता है कि आपको अपने Cloud Firestore डेटाबेस के ऐक्सेस को लॉक करना पड़े. हालांकि, ऐप्लिकेशन लॉन्च करने से पहले, आपको बेहतर तरीके से सेट अप करना होगा Cloud Firestore Security Rules. Cloud Firestore एम्युलेटर की मदद से, अपने ऐप्लिकेशन की सामान्य सुविधाओं और उसके काम करने के तरीके को प्रोटोटाइप और टेस्ट किया जा सकता है. इसके अलावा, यूनिट टेस्ट भी लिखे जा सकते हैं. इनकी मदद से, Cloud Firestore Security Rules के काम करने के तरीके की जांच की जा सकती है.
क्विकस्टार्ट
आसान नियमों वाले कुछ बुनियादी टेस्ट केस के लिए, क्विकस्टार्ट सैंपल आज़माएं.
के बारे में जानकारीCloud Firestore Security Rules
मोबाइल और वेब क्लाइंट लाइब्रेरी का इस्तेमाल करते समय, सर्वरलेस पुष्टि, अनुमति, और डेटा की पुष्टि के लिए, Firebase Authentication और Cloud Firestore Security Rules को लागू करें.
Cloud Firestore Security Rules में ये दो चीज़ें शामिल होती हैं:
- एक
matchस्टेटमेंट, जो आपके डेटाबेस में मौजूद दस्तावेज़ों की पहचान करता है. - एक
allowएक्सप्रेशन, जो उन दस्तावेज़ों के ऐक्सेस को कंट्रोल करता है.
Firebase Authentication उपयोगकर्ताओं के क्रेडेंशियल की पुष्टि करता है. साथ ही, यह उपयोगकर्ता के आधार पर और भूमिका के आधार पर ऐक्सेस सिस्टम के लिए आधार उपलब्ध कराता है.
Cloud Firestore मोबाइल/वेब क्लाइंट लाइब्रेरी से किए गए डेटाबेस के हर अनुरोध का आकलन, कोई भी डेटा पढ़ने या लिखने से पहले, आपके सुरक्षा नियमों के हिसाब से किया जाता है. अगर नियम, तय किए गए किसी भी दस्तावेज़ पाथ के ऐक्सेस को अस्वीकार करते हैं, तो पूरा अनुरोध फ़ेल हो जाता है.
Cloud Firestore Security Rules के बारे में ज़्यादा जानने के लिए, Cloud Firestore Security Rules लेख पढ़ें.
एम्युलेटर इंस्टॉल करना
Cloud Firestore एम्युलेटर इंस्टॉल करने के लिए, Firebase CLI का इस्तेमाल करें और नीचे दिया गया कमांड चलाएं:
firebase setup:emulators:firestore
एम्युलेटर चलाना
शुरू करने के लिए, अपनी वर्किंग डायरेक्ट्री में Firebase प्रोजेक्ट को शुरू करें. Firebase CLI का इस्तेमाल करते समय, यह पहला सामान्य चरण है.
firebase init
एम्युलेटर शुरू करने के लिए, यह कमांड इस्तेमाल करें. एम्युलेटर तब तक चलेगा, जब तक आप प्रोसेस को बंद नहीं कर देते:
firebase emulators:start --only firestore
कई मामलों में, आपको एम्युलेटर शुरू करना, टेस्ट सुइट चलाना, और फिर टेस्ट चलने के बाद एम्युलेटर को बंद करना होता है. emulators:exec कमांड का इस्तेमाल करके, यह काम आसानी से किया जा सकता है:
firebase emulators:exec --only firestore "./my-test-script.sh"
एम्युलेटर शुरू होने पर, यह डिफ़ॉल्ट पोर्ट (8080) पर चलने की कोशिश करेगा. अपनी
firebase.json फ़ाइल के "emulators" सेक्शन में बदलाव करके, एम्युलेटर का पोर्ट बदला जा सकता है:
{
// ...
"emulators": {
"firestore": {
"port": "YOUR_PORT"
}
}
}एम्युलेटर चलाने से पहले
एम्युलेटर का इस्तेमाल शुरू करने से पहले, इन बातों का ध्यान रखें:
- एम्युलेटर, शुरू में आपकी
firebase.jsonफ़ाइल केfirestore.rulesफ़ील्ड में तय किए गए नियमों को लोड करेगा. यह, स्थानीय फ़ाइल के नाम की उम्मीद करता है जिसमें आपके Cloud Firestore Security Rules शामिल हैं. साथ ही, उन नियमों को सभी प्रोजेक्ट पर लागू करता है. अगर स्थानीय फ़ाइल का पाथ नहीं दिया जाता है या नीचे बताए गए तरीके सेloadFirestoreRulesतरीके का इस्तेमाल नहीं किया जाता है, तो एम्युलेटर सभी प्रोजेक्ट को खुले नियमों के तौर पर मानता है. - ज़्यादातर Firebase SDK टूल, एम्युलेटर के साथ सीधे काम करते हैं. हालांकि, सिर्फ़
@firebase/rules-unit-testingलाइब्रेरी, सुरक्षा नियमों मेंauthको मॉक करने की सुविधा देती है. इससे यूनिट टेस्ट करना बहुत आसान हो जाता है. इसके अलावा, लाइब्रेरी, एम्युलेटर की कुछ खास सुविधाओं के साथ काम करती है. जैसे, सभी डेटा को मिटाना. इनकी जानकारी नीचे दी गई है. - एम्युलेटर, क्लाइंट SDK टूल से मिले, प्रोडक्शन Firebase Auth टोकन भी स्वीकार करेंगे. साथ ही, नियमों का आकलन भी करेंगे. इससे, इंटिग्रेशन और मैन्युअल टेस्ट में, अपने ऐप्लिकेशन को सीधे एम्युलेटर से कनेक्ट किया जा सकेगा.
लोकल यूनिट टेस्ट करना
JavaScript SDK टूल के v9 वर्शन के साथ लोकल यूनिट टेस्ट करना
Firebase, सुरक्षा नियमों की यूनिट टेस्टिंग लाइब्रेरी को JavaScript SDK टूल के वर्शन 9 और वर्शन 8, दोनों के साथ डिस्ट्रिब्यूट करता है. लाइब्रेरी के एपीआई में काफ़ी अंतर है. हमारा सुझाव है कि टेस्टिंग लाइब्रेरी के v9 वर्शन का इस्तेमाल करें. यह ज़्यादा बेहतर है. साथ ही, इसे एम्युलेटर से कनेक्ट करने के लिए कम सेटअप की ज़रूरत होती है. इससे, प्रोडक्शन के संसाधनों का गलती से इस्तेमाल होने से बचाया जा सकता है. हम, पुराने सिस्टम के साथ काम करने की सुविधा के लिए, टेस्टिंग लाइब्रेरी के v8 वर्शन को उपलब्ध कराते रहेंगे.
- v9 SDK टूल में, टेस्ट के सामान्य तरीके और यूटिलिटी फ़ंक्शन
- v9 SDK टूल में, एम्युलेटर के लिए खास टेस्ट के तरीके
स्थानीय तौर पर चलने वाले एम्युलेटर के साथ इंटरैक्ट करने के लिए, @firebase/rules-unit-testing मॉड्यूल का इस्तेमाल करें. अगर आपको टाइम आउट या ECONNREFUSED से जुड़ी गड़बड़ियां मिलती हैं, तो पक्का करें कि एम्युलेटर सही तरीके से चल रहा हो.
हमारा सुझाव है कि Node.js के नए वर्शन का इस्तेमाल करें, ताकि async/await नोटेशन का इस्तेमाल किया जा सके. जिस व्यवहार को टेस्ट करना है उसमें ज़्यादातर एसिंक्रोनस फ़ंक्शन शामिल होते हैं. साथ ही, टेस्टिंग मॉड्यूल को प्रॉमिस-आधारित कोड के साथ काम करने के लिए डिज़ाइन किया गया है.
नियमों की यूनिट टेस्टिंग लाइब्रेरी का v9 वर्शन, हमेशा एम्युलेटर के बारे में जानता है. साथ ही, यह आपके प्रोडक्शन के संसाधनों को कभी ऐक्सेस नहीं करता.
लाइब्रेरी को इंपोर्ट करने के लिए, v9 मॉड्यूलर इंपोर्ट स्टेटमेंट का इस्तेमाल करें. उदाहरण के लिए:
import {
assertFails,
assertSucceeds,
initializeTestEnvironment
} from "@firebase/rules-unit-testing"
// Use `const { … } = require("@firebase/rules-unit-testing")` if imports are not supported
// Or we suggest `const testing = require("@firebase/rules-unit-testing")` if necessary.
इंपोर्ट करने के बाद, यूनिट टेस्ट लागू करने के लिए:
initializeTestEnvironmentको कॉल करके,RulesTestEnvironmentबनाना और कॉन्फ़िगर करना.- नियमों को ट्रिगर किए बिना, टेस्ट डेटा सेट अप करना. इसके लिए,
RulesTestEnvironment.withSecurityRulesDisabledका इस्तेमाल करना. यह एक आसान तरीका है, जिससे नियमों को अस्थायी तौर पर बायपास किया जा सकता है. - टेस्ट डेटा और एनवायरमेंट को साफ़ करने के लिए,
RulesTestEnvironment.cleanup()याRulesTestEnvironment.clearFirestore()जैसे कॉल करके, टेस्ट सुइट और हर टेस्ट के लिए, पहले/बाद में हुक सेट अप करना. RulesTestEnvironment.authenticatedContextऔरRulesTestEnvironment.unauthenticatedContextका इस्तेमाल करके, पुष्टि के स्टेटस की नकल करने वाले टेस्ट केस लागू करना.
सामान्य तरीके और यूटिलिटी फ़ंक्शन
v9 SDK टूल में, एम्युलेटर के लिए खास टेस्ट के तरीके भी देखें.
initializeTestEnvironment() => RulesTestEnvironment
यह फ़ंक्शन, नियमों की यूनिट टेस्टिंग के लिए टेस्ट एनवायरमेंट शुरू करता है. टेस्ट सेटअप के लिए, इस फ़ंक्शन को सबसे पहले कॉल करें. इसे सही तरीके से चलाने के लिए, एम्युलेटर का चालू होना ज़रूरी है.
यह फ़ंक्शन, TestEnvironmentConfig को तय करने वाला एक वैकल्पिक ऑब्जेक्ट स्वीकार करता है. इसमें प्रोजेक्ट आईडी और एम्युलेटर कॉन्फ़िगरेशन सेटिंग शामिल हो सकती हैं.
let testEnv = await initializeTestEnvironment({ projectId: "demo-project-1234", firestore: { rules: fs.readFileSync("firestore.rules", "utf8"), }, });
RulesTestEnvironment.authenticatedContext({ user_id: string, tokenOptions?: TokenOptions }) => RulesTestContext
यह तरीका, RulesTestContext बनाता है. यह पुष्टि किए गए Authentication उपयोगकर्ता की तरह काम करता है. दिए गए कॉन्टेक्स्ट के ज़रिए बनाए गए अनुरोधों में, मॉक Authentication टोकन अटैच होगा. ज़रूरत पड़ने पर, कस्टम दावे या Authentication टोकन पेलोड के लिए ओवरराइड तय करने वाला कोई ऑब्जेक्ट पास करें.
कॉन्फ़िगर किए गए किसी भी एम्युलेटर इंस्टेंस को ऐक्सेस करने के लिए, अपने टेस्ट में, दिए गए टेस्ट कॉन्टेक्स्ट ऑब्जेक्ट का इस्तेमाल करें. इनमें वे इंस्टेंस भी शामिल हैं जिन्हें initializeTestEnvironment के साथ कॉन्फ़िगर किया गया है.
// Assuming a Firestore app and the Firestore emulator for this example import { setDoc } from "firebase/firestore"; const alice = testEnv.authenticatedContext("alice", { … }); // Use the Firestore instance associated with this context await assertSucceeds(setDoc(alice.firestore().doc('/users/alice'), { ... });
RulesTestEnvironment.unauthenticatedContext() => RulesTestContext
यह तरीका, RulesTestContext बनाता है. यह ऐसे क्लाइंट की तरह काम करता है जिसने Authentication के ज़रिए लॉग इन नहीं किया है. दिए गए कॉन्टेक्स्ट के ज़रिए बनाए गए अनुरोधों में, Firebase Auth टोकन अटैच नहीं होंगे.
कॉन्फ़िगर किए गए किसी भी एम्युलेटर इंस्टेंस को ऐक्सेस करने के लिए, अपने टेस्ट में, दिए गए टेस्ट कॉन्टेक्स्ट ऑब्जेक्ट का इस्तेमाल करें. इनमें वे इंस्टेंस भी शामिल हैं जिन्हें initializeTestEnvironment के साथ कॉन्फ़िगर किया गया है.
// Assuming a Cloud Storage app and the Storage emulator for this example import { getStorage, ref, deleteObject } from "firebase/storage"; const alice = testEnv.unauthenticatedContext(); // Use the Cloud Storage instance associated with this context const desertRef = ref(alice.storage(), 'images/desert.jpg'); await assertSucceeds(deleteObject(desertRef));
RulesTestEnvironment.withSecurityRulesDisabled()
सुरक्षा नियमों को बंद करने पर, टेस्ट सेटअप फ़ंक्शन को ऐसे कॉन्टेक्स्ट के साथ चलाएं जो सुरक्षा नियमों को बंद करने पर काम करता है.
इस तरीके में एक कॉलबैक फ़ंक्शन होता है. यह सुरक्षा नियमों को बायपास करने वाले कॉन्टेक्स्ट को लेता है और एक प्रॉमिस दिखाता है. प्रॉमिस के रिज़ॉल्व या रिजेक्ट होने के बाद, कॉन्टेक्स्ट मिट जाएगा.
RulesTestEnvironment.cleanup()
यह तरीका, टेस्ट एनवायरमेंट में बनाए गए सभी RulesTestContexts को मिटा देता है. साथ ही, इससे जुड़े संसाधनों को साफ़ कर देता है. इससे, टेस्ट को सही तरीके से बंद किया जा सकता है.
इस तरीके से, एम्युलेटर के स्टेटस में कोई बदलाव नहीं होता. टेस्ट के बीच डेटा रीसेट करने के लिए, ऐप्लिकेशन एम्युलेटर के लिए खास डेटा मिटाने का तरीका इस्तेमाल करें.
assertSucceeds(pr: Promise<any>)) => Promise<any>
यह टेस्ट केस यूटिलिटी फ़ंक्शन है.
यह फ़ंक्शन पुष्टि करता है कि एम्युलेटर की किसी कार्रवाई को रैप करने वाला दिया गया प्रॉमिस, सुरक्षा नियमों के उल्लंघन के बिना रिज़ॉल्व हो जाएगा.
await assertSucceeds(setDoc(alice.firestore(), '/users/alice'), { ... });assertFails(pr: Promise<any>)) => Promise<any>
यह टेस्ट केस यूटिलिटी फ़ंक्शन है.
यह फ़ंक्शन पुष्टि करता है कि एम्युलेटर की किसी कार्रवाई को रैप करने वाला दिया गया प्रॉमिस, सुरक्षा नियमों के उल्लंघन के साथ रिजेक्ट हो जाएगा.
await assertFails(setDoc(alice.firestore(), '/users/bob'), { ... });एम्युलेटर के लिए खास तरीके
v9 SDK टूल में, टेस्ट के सामान्य तरीके और यूटिलिटी फ़ंक्शन भी देखें.
RulesTestEnvironment.clearFirestore() => Promise<void>
यह तरीका, Firestore डेटाबेस में मौजूद उस डेटा को मिटाता है जो Firestore एम्युलेटर के लिए कॉन्फ़िगर किए गए projectId से जुड़ा है.
RulesTestContext.firestore(settings?: Firestore.FirestoreSettings) => Firestore;
यह तरीका, इस टेस्ट कॉन्टेक्स्ट के लिए Firestore इंस्टेंस हासिल करता है. दिए गए Firebase JS क्लाइंट SDK टूल के इंस्टेंस का इस्तेमाल, क्लाइंट SDK टूल के एपीआई (v9 मॉड्यूलर या v9 कंपैट) के साथ किया जा सकता है.
नियमों के आकलन को विज़ुअलाइज़ करना
Cloud Firestore एम्युलेटर की मदद से, Emulator Suite के यूज़र इंटरफ़ेस (यूआई) में क्लाइंट के अनुरोधों को विज़ुअलाइज़ किया जा सकता है. इसमें, Firebase के सुरक्षा नियमों के आकलन की ट्रेसिंग भी शामिल है.
हर अनुरोध के लिए, आकलन का क्रम देखने के लिए, Firestore > अनुरोध टैब खोलें.
टेस्ट रिपोर्ट जनरेट करना
टेस्ट का सुइट चलाने के बाद, टेस्ट कवरेज रिपोर्ट ऐक्सेस की जा सकती हैं. इनमें यह दिखाया जाता है कि आपके हर सुरक्षा नियम का आकलन कैसे किया गया.
रिपोर्ट पाने के लिए, एम्युलेटर के चालू होने के दौरान, उस पर मौजूद किसी एंडपॉइंट के बारे में क्वेरी करें. ब्राउज़र के साथ काम करने वाले वर्शन के लिए, यह यूआरएल इस्तेमाल करें:
http://localhost:8080/emulator/v1/projects/<project_id>:ruleCoverage.html
इससे आपके नियमों को एक्सप्रेशन और सब-एक्सप्रेशन में बांटा जाता है. इन पर कर्सर ले जाकर, ज़्यादा जानकारी देखी जा सकती है. इसमें, आकलनों की संख्या और दिखाई गई वैल्यू शामिल हैं. इस डेटा के रॉ JSON वर्शन के लिए, अपनी क्वेरी में यह यूआरएल शामिल करें:
http://localhost:8080/emulator/v1/projects/<project_id>:ruleCoverage
एम्युलेटर और प्रोडक्शन के बीच अंतर
- आपको Cloud Firestore प्रोजेक्ट साफ़ तौर पर बनाने की ज़रूरत नहीं होती. एम्युलेटर, ऐक्सेस किए गए किसी भी इंस्टेंस को अपने-आप बना लेता है.
- Cloud Firestore एम्युलेटर, Firebase Authentication की सामान्य प्रोसेस के साथ काम नहीं करता.
इसके बजाय, Firebase Test SDK टूल में, हमने
rules-unit-testingलाइब्रेरी मेंinitializeTestApp()तरीका उपलब्ध कराया है. इसमेंauthफ़ील्ड होता है. इस तरीके का इस्तेमाल करके बनाया गया Firebase हैंडल, ऐसा काम करेगा जैसे कि आपने जिस भी इकाई की जानकारी दी है उसकी पुष्टि हो गई हो. अगरnullपास किया जाता है, तो यह पुष्टि नहीं किए गए उपयोगकर्ता की तरह काम करेगा (auth != nullनियम फ़ेल हो जाएंगे, उदाहरण के लिए).
आम तौर पर आने वाली समस्याओं को हल करना
Cloud Firestore एम्युलेटर का इस्तेमाल करते समय, आपको आम तौर पर आने वाली ये समस्याएं हो सकती हैं. अगर आपको कोई समस्या आ रही है, तो उसे हल करने के लिए, यहां दिया गया तरीका अपनाएं. ये नोट, सुरक्षा नियमों की यूनिट टेस्टिंग लाइब्रेरी को ध्यान में रखकर लिखे गए हैं. हालांकि, सामान्य तरीके किसी भी Firebase SDK टूल पर लागू होते हैं.
टेस्ट का व्यवहार एक जैसा नहीं है
अगर आपके टेस्ट कभी पास हो रहे हैं और कभी फ़ेल हो रहे हैं, तब भी जब टेस्ट में कोई बदलाव नहीं किया गया है, तो आपको यह पुष्टि करनी पड़ सकती है कि वे सही क्रम में हैं.
एम्युलेटर के साथ ज़्यादातर इंटरैक्शन एसिंक्रोनस होते हैं. इसलिए, पक्का करें कि सभी एसिंक्रोनस कोड सही क्रम में हों. प्रॉमिस को चेन करके या await नोटेशन का इस्तेमाल करके, क्रम को ठीक किया जा सकता है.
खास तौर पर, इन एसिंक्रोनस कार्रवाइयों की समीक्षा करें:
- सुरक्षा नियम सेट करना. उदाहरण के लिए,
initializeTestEnvironmentका इस्तेमाल करना. - डेटा पढ़ना और लिखना. उदाहरण के लिए,
db.collection("users").doc("alice").get()का इस्तेमाल करना. - ऑपरेशनल दावे, जिनमें
assertSucceedsऔरassertFailsशामिल हैं.
एम्युलेटर लोड करने पर, टेस्ट सिर्फ़ पहली बार पास होते हैं
एम्युलेटर, स्टेटफ़ुल होता है. यह, इसमें लिखा गया सारा डेटा मेमोरी में सेव करता है. इसलिए, एम्युलेटर बंद होने पर सारा डेटा मिट जाता है. अगर एक ही प्रोजेक्ट आईडी के लिए कई टेस्ट किए जा रहे हैं, तो हर टेस्ट ऐसा डेटा जनरेट कर सकता है जिससे आने वाले टेस्ट पर असर पड़ सकता है. इस व्यवहार को बायपास करने के लिए, इनमें से कोई भी तरीका इस्तेमाल किया जा सकता है:
- हर टेस्ट के लिए, यूनीक प्रोजेक्ट आईडी इस्तेमाल करें. ध्यान दें कि अगर ऐसा किया जाता है, तो आपको हर टेस्ट के हिस्से के तौर पर
initializeTestEnvironmentको कॉल करना होगा. नियम सिर्फ़ डिफ़ॉल्ट प्रोजेक्ट आईडी के लिए अपने-आप लोड होते हैं. - अपने टेस्ट को इस तरह से रीस्ट्रक्चर करें कि वे पहले से लिखे गए डेटा के साथ इंटरैक्ट न करें. उदाहरण के लिए, हर टेस्ट के लिए अलग कलेक्शन का इस्तेमाल करें.
- किसी टेस्ट के दौरान लिखा गया सारा डेटा मिटाएं.
टेस्ट सेटअप बहुत मुश्किल है
टेस्ट सेट अप करते समय, हो सकता है कि आपको डेटा में इस तरह से बदलाव करना पड़े जिसकी अनुमति आपके
Cloud Firestore Security Rules से न मिलती हो. अगर आपके नियमों की वजह से टेस्ट सेटअप मुश्किल हो रहा है, तो सेटअप के चरणों में RulesTestEnvironment.withSecurityRulesDisabled का इस्तेमाल करें. इससे, पढ़ने और लिखने की कार्रवाइयों से PERMISSION_DENIED से जुड़ी गड़बड़ियां ट्रिगर नहीं होंगी.
इसके बाद, आपका टेस्ट, RulesTestEnvironment.authenticatedContext और unauthenticatedContext का इस्तेमाल करके, पुष्टि किए गए या पुष्टि नहीं किए गए उपयोगकर्ता के तौर पर कार्रवाइयां कर सकता है. इससे, यह पुष्टि की जा सकती है कि आपके Cloud Firestore Security Rules अलग-अलग मामलों में सही तरीके से अनुमति देते हैं या अस्वीकार करते हैं.