1. بررسی اجمالی
اهداف
در این کد لبه شما یک برنامه توصیه رستوران با پشتیبانی Firestor را در iOS در سوئیفت خواهید ساخت. شما یاد خواهید گرفت که چگونه:
- خواندن و نوشتن داده ها در Firestore از یک برنامه iOS
- به تغییرات داده های Firestore در زمان واقعی گوش دهید
- از Firebase Authentication و قوانین امنیتی برای ایمن سازی داده های Firestore استفاده کنید
- پرس و جوهای پیچیده Firestore را بنویسید
پیش نیازها
قبل از شروع این کد لبه مطمئن شوید که نصب کرده اید:
- Xcode نسخه 14.0 (یا بالاتر)
- CocoaPods 1.12.0 (یا بالاتر)
2. پروژه کنسول Firebase را ایجاد کنید
Firebase را به پروژه اضافه کنید
- به کنسول Firebase بروید.
- Create New Project را انتخاب کنید و نام پروژه خود را "Firestore iOS Codelab" بگذارید.
3. نمونه پروژه را دریافت کنید
کد را دانلود کنید
با شبیه سازی پروژه نمونه و اجرای pod update
در فهرست پروژه شروع کنید:
git clone https://github.com/firebase/friendlyeats-ios cd friendlyeats-ios pod update
FriendlyEats.xcworkspace
را در Xcode باز کنید و آن را اجرا کنید (Cmd+R). برنامه باید به درستی کامپایل شود و بلافاصله پس از راه اندازی خراب شود، زیرا یک فایل GoogleService-Info.plist
در آن وجود ندارد. در مرحله بعد آن را اصلاح خواهیم کرد.
Firebase را راه اندازی کنید
برای ایجاد یک پروژه Firestore جدید ، مستندات را دنبال کنید. هنگامی که پروژه خود را دریافت کردید، فایل GoogleService-Info.plist
پروژه خود را از کنسول Firebase دانلود کنید و آن را به ریشه پروژه Xcode بکشید. پروژه را دوباره اجرا کنید تا مطمئن شوید که برنامه به درستی پیکربندی شده و دیگر هنگام راه اندازی خراب نمی شود. پس از ورود به سیستم، باید یک صفحه خالی مانند مثال زیر مشاهده کنید. اگر نمیتوانید وارد سیستم شوید، مطمئن شوید که روش ورود به سیستم ایمیل/گذرواژه را در کنسول Firebase در قسمت Authentication فعال کردهاید.
4. داده ها را در Firestore بنویسید
در این بخش دادههایی را در Firestore مینویسیم تا بتوانیم رابط کاربری برنامه را پر کنیم. این را می توان به صورت دستی از طریق کنسول Firebase انجام داد، اما ما این کار را در خود برنامه انجام خواهیم داد تا یک نوشتن اولیه Firestore را نشان دهیم.
شی مدل اصلی در برنامه ما یک رستوران است. داده های Firestore به اسناد، مجموعه ها و زیر مجموعه ها تقسیم می شوند. ما هر رستوران را به عنوان یک سند در یک مجموعه سطح بالا به نام restaurants
ذخیره می کنیم. اگر میخواهید درباره مدل داده Firestore اطلاعات بیشتری کسب کنید، درباره اسناد و مجموعهها در مستندات مطالعه کنید.
قبل از اینکه بتوانیم دادهها را به Firestore اضافه کنیم، باید به مجموعه رستورانها اشاره کنیم. موارد زیر را به حلقه for داخلی در متد RestaurantsTableViewController.didTapPopulateButton(_:)
اضافه کنید.
let collection = Firestore.firestore().collection("restaurants")
اکنون که یک مرجع مجموعه داریم، می توانیم برخی از داده ها را بنویسیم. درست بعد از آخرین خط کدی که اضافه کردیم موارد زیر را اضافه کنید:
let collection = Firestore.firestore().collection("restaurants")
// ====== ADD THIS ======
let restaurant = Restaurant(
name: name,
category: category,
city: city,
price: price,
ratingCount: 0,
averageRating: 0
)
collection.addDocument(data: restaurant.dictionary)
کد بالا یک سند جدید به مجموعه رستوران ها اضافه می کند. دادههای سند از یک فرهنگ لغت میآیند که ما از ساختار رستوران دریافت میکنیم.
ما تقریباً به آنجا رسیده ایم – قبل از اینکه بتوانیم اسنادی را در Firestore بنویسیم، باید قوانین امنیتی Firestore را باز کنیم و توضیح دهیم که کدام بخش از پایگاه داده ما باید توسط کدام کاربر قابل نوشتن باشد. در حال حاضر، ما فقط به کاربران تأیید شده اجازه میدهیم تا در کل پایگاه داده بخوانند و بنویسند. این برای یک برنامه تولیدی کمی بیش از حد مجاز است، اما در طول فرآیند ساخت اپلیکیشن، ما به اندازه کافی آرامشبخش میخواهیم تا در حین آزمایش دائماً با مشکلات احراز هویت مواجه نشویم. در پایان این کد، در مورد چگونگی سختتر کردن قوانین امنیتی و محدود کردن امکان خواندن و نوشتن ناخواسته صحبت خواهیم کرد.
در برگه قوانین کنسول Firebase قوانین زیر را اضافه کنید و سپس روی انتشار کلیک کنید.
rules_version = '2'; service cloud.firestore { match /databases/{database}/documents { match /{document=**} { // // WARNING: These rules are insecure! We will replace them with // more secure rules later in the codelab // allow read, write: if request.auth != null; } } }
قوانین امنیتی را بعداً به تفصیل مورد بحث قرار خواهیم داد، اما اگر عجله دارید، نگاهی به مستندات قوانین امنیتی بیندازید.
برنامه را اجرا کرده و وارد شوید. سپس روی دکمه « پر کردن » در بالا سمت چپ ضربه بزنید، که دستهای از اسناد رستوران ایجاد میکند، اگرچه هنوز این مورد را در برنامه مشاهده نخواهید کرد.
سپس به برگه داده Firestore در کنسول Firebase بروید. اکنون باید ورودی های جدید را در مجموعه رستوران ها مشاهده کنید:
تبریک میگوییم، شما به تازگی دادههایی را از یک برنامه iOS برای Firestore نوشتهاید! در بخش بعدی نحوه بازیابی داده ها از Firestore و نمایش آن ها در برنامه را خواهید آموخت.
5. نمایش داده ها از Firestore
در این بخش با نحوه بازیابی اطلاعات از Firestore و نمایش آن در اپلیکیشن آشنا می شوید. دو مرحله کلیدی ایجاد یک پرس و جو و افزودن یک شنونده عکس فوری است. این شنونده از تمام داده های موجود که با پرس و جو مطابقت دارد مطلع می شود و به روز رسانی ها را در زمان واقعی دریافت می کند.
ابتدا، بیایید پرس و جوی را بسازیم که فهرست پیشفرض و فیلتر نشده رستورانها را ارائه کند. نگاهی به اجرای RestaurantsTableViewController.baseQuery()
بیندازید:
return Firestore.firestore().collection("restaurants").limit(to: 50)
این پرس و جو تا 50 رستوران از مجموعه سطح بالا به نام "رستوران" را بازیابی می کند. اکنون که یک پرس و جو داریم، باید یک شنونده عکس فوری را برای بارگذاری داده ها از Firestore در برنامه خود ضمیمه کنیم. کد زیر را دقیقاً پس از فراخوانی stopObserving()
به متد RestaurantsTableViewController.observeQuery()
اضافه کنید.
listener = query.addSnapshotListener { [unowned self] (snapshot, error) in
guard let snapshot = snapshot else {
print("Error fetching snapshot results: \(error!)")
return
}
let models = snapshot.documents.map { (document) -> Restaurant in
if let model = Restaurant(dictionary: document.data()) {
return model
} else {
// Don't use fatalError here in a real app.
fatalError("Unable to initialize type \(Restaurant.self) with dictionary \(document.data())")
}
}
self.restaurants = models
self.documents = snapshot.documents
if self.documents.count > 0 {
self.tableView.backgroundView = nil
} else {
self.tableView.backgroundView = self.backgroundView
}
self.tableView.reloadData()
}
کد بالا مجموعه را از Firestore دانلود کرده و در یک آرایه به صورت محلی ذخیره می کند. فراخوانی addSnapshotListener(_:)
یک شنونده عکس فوری به کوئری اضافه می کند که هر بار که داده ها در سرور تغییر می کنند، view controller را به روز می کند. ما بهروزرسانیها را بهطور خودکار دریافت میکنیم و نیازی نیست تغییرات را بهصورت دستی انجام دهیم. به یاد داشته باشید، این شنونده عکس فوری را می توان در هر زمانی در نتیجه تغییر سمت سرور فراخوانی کرد، بنابراین مهم است که برنامه ما بتواند تغییرات را مدیریت کند.
پس از نگاشت دیکشنریهای ما به ساختارها (به Restaurant.swift
مراجعه کنید)، نمایش دادهها فقط با اختصاص دادن چند ویژگی view است. خطوط زیر را به RestaurantTableViewCell.populate(restaurant:)
در RestaurantsTableViewController.swift
اضافه کنید.
nameLabel.text = restaurant.name
cityLabel.text = restaurant.city
categoryLabel.text = restaurant.category
starsView.rating = Int(restaurant.averageRating.rounded())
priceLabel.text = priceString(from: restaurant.price)
این متد جمعآوری از طریق متد tableView tableView(_:cellForRowAtIndexPath:)
منبع داده جدول نمای فراخوانی میشود، که از نگاشت مجموعه انواع مقادیر از قبل به سلولهای نمای جدول جداگانه مراقبت میکند.
برنامه را دوباره اجرا کنید و بررسی کنید که رستوران هایی که قبلاً در کنسول دیده بودیم اکنون در شبیه ساز یا دستگاه قابل مشاهده هستند. اگر این بخش را با موفقیت کامل کردید، اکنون برنامه شما در حال خواندن و نوشتن داده ها با Cloud Firestore است!
6. مرتب سازی و فیلتر کردن داده ها
در حال حاضر برنامه ما لیستی از رستوران ها را نمایش می دهد، اما هیچ راهی برای فیلتر کردن کاربر بر اساس نیازهای خود وجود ندارد. در این بخش از جستجوی پیشرفته Firestore برای فعال کردن فیلتر کردن استفاده خواهید کرد.
در اینجا نمونه ای از یک پرس و جو ساده برای واکشی همه رستوران های دیم سام آورده شده است:
let filteredQuery = query.whereField("category", isEqualTo: "Dim Sum")
همانطور که از نامش پیداست، متد whereField(_:isEqualTo:)
باعث میشود که درخواست ما فقط اعضای مجموعهای را دانلود کند که فیلدهایشان با محدودیتهایی که ما تعیین کردهایم را برآورده میکند. در این صورت، فقط رستورانهایی را دانلود میکند که category
"Dim Sum"
است.
در این برنامه کاربر می تواند چندین فیلتر را برای ایجاد پرس و جوهای خاص مانند "پیتزا در سانفرانسیسکو" یا "غذاهای دریایی در لس آنجلس سفارش داده شده توسط محبوبیت" زنجیره ای کند.
RestaurantsTableViewController.swift
را باز کنید و بلوک کد زیر را به وسط query(withCategory:city:price:sortBy:)
:
if let category = category, !category.isEmpty {
filtered = filtered.whereField("category", isEqualTo: category)
}
if let city = city, !city.isEmpty {
filtered = filtered.whereField("city", isEqualTo: city)
}
if let price = price {
filtered = filtered.whereField("price", isEqualTo: price)
}
if let sortBy = sortBy, !sortBy.isEmpty {
filtered = filtered.order(by: sortBy)
}
قطعه بالا چندین عبارت whereField
و order
را برای ساخت یک کوئری ترکیبی واحد بر اساس ورودی کاربر اضافه می کند. اکنون درخواست ما فقط رستورانهایی را برمیگرداند که با نیازهای کاربر مطابقت دارند.
پروژه خود را اجرا کنید و تأیید کنید که میتوانید براساس قیمت، شهر و دسته فیلتر کنید (حتماً نام دسته و شهر را دقیقاً تایپ کنید). در حین آزمایش، ممکن است خطاهایی در گزارش های خود مشاهده کنید که به شکل زیر است:
Error fetching snapshot results: Error Domain=io.grpc Code=9 "The query requires an index. You can create it here: https://console.firebase.google.com/project/project-id/database/firestore/indexes?create_composite=..." UserInfo={NSLocalizedDescription=The query requires an index. You can create it here: https://console.firebase.google.com/project/project-id/database/firestore/indexes?create_composite=...}
این به این دلیل است که Firestore برای اکثر پرس و جوهای ترکیبی به ایندکس نیاز دارد. نیاز به نمایهها در کوئریها Firestore را در مقیاس سریع نگه میدارد. با باز کردن پیوند از پیام خطا، رابط کاربری ایجاد فهرست به طور خودکار در کنسول Firebase با پارامترهای صحیح پر شده باز می شود. برای کسب اطلاعات بیشتر در مورد نمایه ها در Firestore، از مستندات دیدن کنید .
7. نوشتن داده در تراکنش
در این بخش، این امکان را برای کاربران اضافه می کنیم که نظرات خود را به رستوران ها ارسال کنند. تا اینجا، همه نوشته های ما اتمی و نسبتاً ساده بوده اند. اگر هر یک از آنها خطا داشت، احتمالاً فقط از کاربر می خواهیم آنها را دوباره امتحان کند یا آنها را دوباره به طور خودکار امتحان کند.
برای افزودن امتیاز به یک رستوران، باید چندین خواندن و نوشتن را هماهنگ کنیم. ابتدا باید خود بررسی ارسال شود و سپس تعداد رتبهبندی رستوران و میانگین رتبهبندی باید بهروزرسانی شود. اگر یکی از این ها خراب شود اما دیگری نه، در وضعیت ناسازگاری قرار می گیریم که در آن داده های یک قسمت از پایگاه داده ما با داده های قسمت دیگر مطابقت ندارد.
خوشبختانه، Firestore عملکرد تراکنشهایی را ارائه میکند که به ما امکان میدهد چندین خواندن و نوشتن را در یک عملیات اتمی انجام دهیم و اطمینان حاصل کنیم که دادههای ما ثابت میمانند.
کد زیر را در زیر تمام اعلانهای let در RestaurantDetailViewController.reviewController(_:didSubmitFormWithReview:)
اضافه کنید.
let firestore = Firestore.firestore()
firestore.runTransaction({ (transaction, errorPointer) -> Any? in
// Read data from Firestore inside the transaction, so we don't accidentally
// update using stale client data. Error if we're unable to read here.
let restaurantSnapshot: DocumentSnapshot
do {
try restaurantSnapshot = transaction.getDocument(reference)
} catch let error as NSError {
errorPointer?.pointee = error
return nil
}
// Error if the restaurant data in Firestore has somehow changed or is malformed.
guard let data = restaurantSnapshot.data(),
let restaurant = Restaurant(dictionary: data) else {
let error = NSError(domain: "FireEatsErrorDomain", code: 0, userInfo: [
NSLocalizedDescriptionKey: "Unable to write to restaurant at Firestore path: \(reference.path)"
])
errorPointer?.pointee = error
return nil
}
// Update the restaurant's rating and rating count and post the new review at the
// same time.
let newAverage = (Float(restaurant.ratingCount) * restaurant.averageRating + Float(review.rating))
/ Float(restaurant.ratingCount + 1)
transaction.setData(review.dictionary, forDocument: newReviewReference)
transaction.updateData([
"numRatings": restaurant.ratingCount + 1,
"avgRating": newAverage
], forDocument: reference)
return nil
}) { (object, error) in
if let error = error {
print(error)
} else {
// Pop the review controller on success
if self.navigationController?.topViewController?.isKind(of: NewReviewViewController.self) ?? false {
self.navigationController?.popViewController(animated: true)
}
}
}
در داخل بلوک بهروزرسانی، تمام عملیاتی که با استفاده از شی تراکنش انجام میدهیم، توسط Firestore به عنوان یک بهروزرسانی اتمی واحد در نظر گرفته میشوند. اگر به روز رسانی روی سرور انجام نشد، Firestore به طور خودکار آن را چند بار دوباره امتحان می کند. این بدان معناست که شرایط خطای ما به احتمال زیاد یک خطای واحد است که به طور مکرر رخ می دهد، به عنوان مثال اگر دستگاه کاملا آفلاین باشد یا کاربر مجاز نباشد در مسیری که می خواهد بنویسد بنویسد.
8. قوانین امنیتی
کاربران برنامه ما نباید قادر به خواندن و نوشتن هر قطعه داده در پایگاه داده ما باشند. به عنوان مثال، همه باید بتوانند رتبهبندیهای یک رستوران را ببینند، اما فقط یک کاربر تأیید شده باید اجازه ارسال رتبهبندی را داشته باشد. نوشتن کد خوب روی کلاینت کافی نیست، ما باید مدل امنیت داده خود را در backend مشخص کنیم تا کاملاً ایمن باشد. در این بخش نحوه استفاده از قوانین امنیتی Firebase را برای محافظت از داده های خود یاد خواهیم گرفت.
ابتدا، بیایید نگاهی عمیقتر به قوانین امنیتی که در ابتدای برنامه کد نوشتیم بیندازیم. کنسول Firebase را باز کنید و به Database > Rules در تب Firestore بروید.
rules_version = '2';
service cloud.firestore {
match /databases/{database}/documents {
match /{document=**} {
// Only authenticated users can read or write data
allow read, write: if request.auth != null;
}
}
}
متغیر request
در قوانین بالا یک متغیر سراسری است که در همه قوانین موجود است، و شرطی که اضافه کردیم تضمین میکند که درخواست قبل از اینکه به کاربران اجازه انجام کاری را بدهد، احراز هویت میشود. این باعث میشود که کاربران احراز هویت نشده از Firestore API برای ایجاد تغییرات غیرمجاز در دادههای شما استفاده کنند. این شروع خوبی است، اما ما می توانیم از قوانین Firestore برای انجام کارهای بسیار قدرتمندتر استفاده کنیم.
اجازه دهید نوشتن مرور را محدود کنیم تا شناسه کاربری مرور باید با شناسه کاربر احراز هویت شده مطابقت داشته باشد. این تضمین می کند که کاربران نمی توانند جعل هویت یکدیگر باشند و نظرات جعلی را از خود به جای بگذارند. قوانین امنیتی خود را با موارد زیر جایگزین کنید:
rules_version = '2';
service cloud.firestore {
match /databases/{database}/documents {
match /restaurants/{any}/ratings/{rating} {
// Users can only write ratings with their user ID
allow read;
allow write: if request.auth != null
&& request.auth.uid == request.resource.data.userId;
}
match /restaurants/{any} {
// Only authenticated users can read or write data
allow read, write: if request.auth != null;
}
}
}
اولین بیانیه تطابق با مجموعه فرعی به نام ratings
هر سند متعلق به مجموعه restaurants
مطابقت دارد. در صورتی که شناسه کاربری مرور با شناسه کاربر مطابقت نداشته باشد، allow write
مشروط از ارسال هرگونه بررسی جلوگیری میکند. عبارت تطبیق دوم به هر کاربر احراز هویت شده اجازه می دهد تا رستوران ها را در پایگاه داده بخواند و بنویسد.
این برای بررسیهای ما بسیار خوب عمل میکند، زیرا ما از قوانین امنیتی برای بیان صریح ضمانت ضمنی که قبلاً در برنامه خود نوشتیم استفاده کردهایم – که کاربران فقط میتوانند نظرات خود را بنویسند. اگر بخواهیم یک تابع ویرایش یا حذف را برای بررسی ها اضافه کنیم، دقیقاً همین مجموعه قوانین همچنین از اصلاح یا حذف نظرات کاربران دیگر توسط کاربران جلوگیری می کند. اما قوانین Firestore همچنین می توانند به شکلی دقیق تر برای محدود کردن نوشتن در فیلدهای جداگانه در اسناد به جای خود اسناد استفاده شوند. میتوانیم از این استفاده کنیم تا به کاربران اجازه دهیم فقط رتبهبندیها، میانگین رتبهبندی و تعداد رتبهبندیهای یک رستوران را بهروزرسانی کنند، و امکان تغییر نام یا مکان رستوران توسط کاربر مخرب را از بین ببریم.
rules_version = '2';
service cloud.firestore {
match /databases/{database}/documents {
match /restaurants/{restaurant} {
match /ratings/{rating} {
allow read: if request.auth != null;
allow write: if request.auth != null
&& request.auth.uid == request.resource.data.userId;
}
allow read: if request.auth != null;
allow create: if request.auth != null;
allow update: if request.auth != null
&& request.resource.data.name == resource.data.name
&& request.resource.data.city == resource.data.city
&& request.resource.data.price == resource.data.price
&& request.resource.data.category == resource.data.category;
}
}
}
در اینجا ما مجوز نوشتن خود را به ایجاد و به روز رسانی تقسیم کرده ایم تا بتوانیم در مورد اینکه کدام عملیات باید مجاز باشد، دقیق تر باشیم. هر کاربری میتواند رستورانها را در پایگاه داده بنویسد، با حفظ عملکرد دکمه Populate که در ابتدای کد لبه درست کردیم، اما وقتی رستورانی نوشته شد نام، مکان، قیمت و دستهبندی آن قابل تغییر نیست. به طور خاص تر، آخرین قانون هر عملیات به روز رسانی رستوران را ملزم می کند که همان نام، شهر، قیمت و دسته فیلدهای موجود در پایگاه داده را حفظ کند.
برای اطلاعات بیشتر در مورد کارهایی که می توانید با قوانین امنیتی انجام دهید، به مستندات نگاهی بیندازید.
9. نتیجه گیری
در این کد لبه، نحوه خواندن و نوشتن مقدماتی و پیشرفته با Firestore و همچنین نحوه ایمن سازی دسترسی به داده ها با قوانین امنیتی را یاد گرفتید. می توانید راه حل کامل را در شاخه codelab-complete
پیدا کنید.
برای کسب اطلاعات بیشتر در مورد Firestor، به منابع زیر مراجعه کنید: