برنامههایی که از توابع نسل اول استفاده میکنند باید با استفاده از دستورالعملهای این راهنما، مهاجرت به نسل دوم را در نظر بگیرند. توابع نسل دوم از Cloud Run برای ارائه عملکرد بهتر، پیکربندی بهتر، نظارت بهتر و موارد دیگر استفاده میکنند.
مثالهای این صفحه فرض میکنند که شما از جاوا اسکریپت با ماژولهای CommonJS ( require
وارد کردن استایل) استفاده میکنید، اما همین اصول برای جاوا اسکریپت با ESM ( import … from
وارد کردن استایل) و TypeScript نیز صدق میکند.
روند مهاجرت
توابع نسل اول و نسل دوم میتوانند در کنار هم در یک فایل وجود داشته باشند. این امر امکان مهاجرت آسان قطعه به قطعه را فراهم میکند، همانطور که آماده هستید. توصیه میکنیم هر بار یک تابع را مهاجرت دهید و قبل از ادامه، آزمایش و تأیید را انجام دهید.
نسخههای Firebase CLI و firebase-function
را تأیید کنید
مطمئن شوید که حداقل از Firebase CLI نسخه 12.00
و firebase-functions
نسخه 4.3.0
استفاده میکنید. هر نسخه جدیدتری از نسل دوم و همچنین نسل اول پشتیبانی خواهد کرد.
بهروزرسانی واردات
توابع نسل دوم از زیربسته v2
در firebase-functions
SDK وارد میشوند. این مسیر واردات متفاوت، تمام چیزی است که Firebase CLI برای تعیین اینکه آیا کد تابع شما را به عنوان یک تابع نسل اول یا دوم مستقر کند، نیاز دارد.
زیربستهی نسخه v2
ماژولار است و توصیه میکنیم فقط ماژول خاصی را که نیاز دارید وارد کنید.
قبل از: نسل اول
const functions = require("firebase-functions/v1");
بعد از: نسل دوم
// explicitly import each trigger
const {onRequest} = require("firebase-functions/v2/https");
const {onDocumentCreated} = require("firebase-functions/v2/firestore");
بهروزرسانی تعاریف تریگر
از آنجایی که نسل دوم SDK از واردات ماژولار پشتیبانی میکند، تعاریف تریگر را بهروزرسانی کنید تا واردات تغییر یافته از مرحله قبل را منعکس کند.
آرگومانهای ارسالی به callbackها برای برخی از triggerها تغییر کردهاند. در این مثال، توجه داشته باشید که آرگومانهای ارسالی به callback onDocumentCreated
در یک شیء event
واحد تجمیع شدهاند. علاوه بر این، برخی از triggerها دارای ویژگیهای پیکربندی جدید و مناسبی هستند، مانند گزینه cors
در trigger مربوط به onRequest
.
قبل از: نسل اول
const functions = require("firebase-functions/v1");
exports.date = functions.https.onRequest((req, res) => {
// ...
});
exports.uppercase = functions.firestore
.document("my-collection/{docId}")
.onCreate((change, context) => {
// ...
});
بعد از: نسل دوم
const {onRequest} = require("firebase-functions/v2/https");
const {onDocumentCreated} = require("firebase-functions/v2/firestore");
exports.date = onRequest({cors: true}, (req, res) => {
// ...
});
exports.uppercase = onDocumentCreated("my-collection/{docId}", (event) => {
/* ... */
});
استفاده از پیکربندی پارامتری
توابع نسل دوم پشتیبانی از functions.config
را کنار گذاشتهاند و به جای آن از یک رابط امنتر برای تعریف پارامترهای پیکربندی به صورت اعلانی در داخل کدبیس شما استفاده میکنند. با ماژول جدید params
، رابط خط فرمان (CLI) استقرار را مسدود میکند، مگر اینکه همه پارامترها مقدار معتبری داشته باشند و تضمین میکند که تابعی با پیکربندی از دست رفته مستقر نشود.
مهاجرت به زیربسته params
اگر از پیکربندی محیط با functions.config
استفاده کردهاید، میتوانید پیکربندی موجود خود را با تغییر ساختار آن به عنوان پیکربندی پارامتری ، منتقل کنید. برای مثال:
قبل از: نسل اول
const functions = require("firebase-functions/v1");
exports.date = functions.https.onRequest((req, res) => {
const date = new Date();
const formattedDate =
date.toLocaleDateString(functions.config().dateformat);
// ...
});
بعد از: نسل دوم
const {onRequest} = require("firebase-functions/v2/https");
const {defineString} = require("firebase-functions/params");
const dateFormat = defineString("DATE_FORMAT");
exports.date = onRequest((req, res) => {
const date = new Date();
const formattedDate = date.toLocaleDateString(dateFormat.value());
// ...
});
تنظیم مقادیر پارامترها
اولین باری که برنامه را مستقر میکنید، رابط خط فرمان فایربیس (Firebase CLI) از شما میخواهد که تمام مقادیر پارامترها را وارد کنید و آنها را در یک فایل dotenv ذخیره کنید. برای خروجی گرفتن از مقادیر functions.config
، دستور firebase functions:config:export
را اجرا کنید.
برای ایمنی بیشتر، میتوانید انواع پارامترها و قوانین اعتبارسنجی را نیز مشخص کنید.
مورد خاص: کلیدهای API
ماژول params
با Cloud Secret Manager ادغام میشود که کنترل دسترسی دقیقی را برای مقادیر حساس مانند کلیدهای API فراهم میکند. برای اطلاعات بیشتر به secret parameters مراجعه کنید.
قبل از: نسل اول
const functions = require("firebase-functions/v1");
exports.getQuote = functions.https.onRequest(async (req, res) => {
const quote = await fetchMotivationalQuote(functions.config().apiKey);
// ...
});
بعد از: نسل دوم
const {onRequest} = require("firebase-functions/v2/https");
const {defineSecret} = require("firebase-functions/params");
// Define the secret parameter
const apiKey = defineSecret("API_KEY");
exports.getQuote = onRequest(
// make the secret available to this function
{ secrets: [apiKey] },
async (req, res) => {
// retrieve the value of the secret
const quote = await fetchMotivationalQuote(apiKey.value());
// ...
}
);
تنظیم گزینههای زمان اجرا
پیکربندی گزینههای زمان اجرا بین نسل اول و دوم تغییر کرده است. نسل دوم همچنین قابلیت جدیدی برای تنظیم گزینهها برای همه عملکردها اضافه کرده است.
قبل از: نسل اول
const functions = require("firebase-functions/v1");
exports.date = functions
.runWith({
// Keep 5 instances warm for this latency-critical function
minInstances: 5,
})
// locate function closest to users
.region("asia-northeast1")
.https.onRequest((req, res) => {
// ...
});
exports.uppercase = functions
// locate function closest to users and database
.region("asia-northeast1")
.firestore.document("my-collection/{docId}")
.onCreate((change, context) => {
// ...
});
بعد از: نسل دوم
const {onRequest} = require("firebase-functions/v2/https");
const {onDocumentCreated} = require("firebase-functions/v2/firestore");
const {setGlobalOptions} = require("firebase-functions/v2");
// locate all functions closest to users
setGlobalOptions({ region: "asia-northeast1" });
exports.date = onRequest({
// Keep 5 instances warm for this latency-critical function
minInstances: 5,
}, (req, res) => {
// ...
});
exports.uppercase = onDocumentCreated("my-collection/{docId}", (event) => {
/* ... */
});
بهروزرسانی حساب سرویس پیشفرض (اختیاری)
در حالی که توابع نسل اول از حساب سرویس پیشفرض موتور برنامه گوگل برای تأیید دسترسی به APIهای Firebase استفاده میکنند، توابع نسل دوم از حساب سرویس پیشفرض موتور محاسبه (Compute Engine) استفاده میکنند. این تفاوت میتواند منجر به مشکلات مجوز برای توابعی شود که به نسل دوم منتقل شدهاند، در مواردی که مجوزهای ویژهای به حساب سرویس نسل اول اعطا کردهاید. اگر مجوزهای هیچ حساب سرویسی را تغییر ندادهاید، میتوانید از این مرحله صرف نظر کنید.
راه حل پیشنهادی این است که حساب سرویس پیشفرض موجود در نسل اول App Engine را به توابعی که میخواهید به نسل دوم منتقل کنید، صریحاً اختصاص دهید و پیشفرض نسل دوم را لغو کنید. میتوانید این کار را با اطمینان از اینکه هر تابع منتقل شده مقدار صحیحی را برای serviceAccountEmail
تنظیم میکند، انجام دهید:
const {onRequest} = require("firebase-functions/https");
const {onDocumentCreated} = require("firebase-functions/v2/firestore");
const {setGlobalOptions} = require("firebase-functions");
// Use the App Engine default service account for all functions
setGlobalOptions({serviceAccountEmail: '<my-project-number>@<wbr>appspot.gserviceaccount.com'});
// Now I use the App Engine default service account.
exports.date = onRequest({cors: true}, (req, res) => {
// ...
});
// I do too!
exports.uppercase = onDocumentCreated("my-collection/{docId}", (event) => {
// ...
});
از طرف دیگر، میتوانید مطمئن شوید که جزئیات حساب سرویس را تغییر میدهید تا با تمام مجوزهای لازم در هر دو حساب سرویس پیشفرض App Engine (برای نسل اول) و حساب سرویس پیشفرض Compute Engine (برای نسل دوم) مطابقت داشته باشد.
استفاده از همزمانی
یکی از مزایای قابل توجه توابع نسل دوم، توانایی یک نمونه تابع واحد برای ارائه بیش از یک درخواست به طور همزمان است. این امر میتواند تعداد شروعهای سرد تجربه شده توسط کاربران نهایی را به طرز چشمگیری کاهش دهد. به طور پیشفرض، همزمانی روی ۸۰ تنظیم شده است، اما میتوانید آن را روی هر مقداری از ۱ تا ۱۰۰۰ تنظیم کنید:
const {onRequest} = require("firebase-functions/v2/https");
exports.date = onRequest({
// set concurrency value
concurrency: 500
},
(req, res) => {
// ...
});
تنظیم همزمانی میتواند عملکرد را بهبود بخشد و هزینه توابع را کاهش دهد. برای اطلاعات بیشتر در مورد همزمانی به Allow concurrent requests مراجعه کنید .
حسابرسی استفاده از متغیرهای سراسری
توابع نسل اول که بدون در نظر گرفتن همزمانی نوشته شدهاند، ممکن است از متغیرهای سراسری استفاده کنند که در هر درخواست تنظیم و خوانده میشوند. وقتی همزمانی فعال باشد و یک نمونه واحد شروع به مدیریت چندین درخواست به طور همزمان کند، این ممکن است باعث ایجاد اشکالاتی در تابع شما شود زیرا درخواستهای همزمان شروع به تنظیم و خواندن متغیرهای سراسری به طور همزمان میکنند.
هنگام ارتقا، میتوانید CPU تابع خود را روی gcf_gen1
تنظیم کنید و concurrency
روی ۱ تنظیم کنید تا رفتار نسل اول بازیابی شود:
const {onRequest} = require("firebase-functions/v2/https");
exports.date = onRequest({
// TEMPORARY FIX: remove concurrency
cpu: "gcf_gen1",
concurrency: 1
},
(req, res) => {
// ...
});
با این حال، این به عنوان یک راه حل بلندمدت توصیه نمیشود، زیرا مزایای عملکرد توابع نسل دوم را از دست میدهد. در عوض، استفاده از متغیرهای سراسری را در توابع خود بررسی کنید و وقتی آماده بودید، این تنظیمات موقت را حذف کنید.
انتقال ترافیک به توابع نسل دوم جدید
درست مانند زمانی که ناحیه یا نوع تریگر یک تابع را تغییر میدهید ، باید به تابع نسل دوم نام جدیدی بدهید و به آرامی ترافیک را به آن منتقل کنید.
نمیتوان یک تابع را از نسل اول به نسل دوم با همان نام ارتقا داد و همزمان firebase deploy
اجرا کرد. انجام این کار منجر به خطای زیر خواهد شد:
Upgrading from GCFv1 to GCFv2 is not yet supported. Please delete your old function or wait for this feature to be ready.
قبل از اینکه این مراحل را دنبال کنید، ابتدا مطمئن شوید که تابع شما idempotent است، زیرا هم نسخه جدید و هم نسخه قدیمی تابع شما در طول تغییر همزمان اجرا میشوند. به عنوان مثال، اگر یک تابع نسل اول دارید که به رویدادهای نوشتن در Firestore پاسخ میدهد، مطمئن شوید که پاسخ به یک نوشتن دو بار، یک بار توسط تابع نسل اول و یک بار توسط تابع نسل دوم، در پاسخ به آن رویدادها، برنامه شما را در حالت ثابتی قرار میدهد.
- نام تابع را در کد توابع خود تغییر دهید. برای مثال،
resizeImage
بهresizeImageSecondGen
تغییر نام دهید. - تابع را مستقر کنید، به طوری که هم تابع نسل اول اصلی و هم تابع نسل دوم در حال اجرا باشند.
- در مورد تریگرهای قابل فراخوانی، صف وظایف و HTTP، با بهروزرسانی کد کلاینت با نام یا URL تابع نسل دوم، شروع به ارجاع همه کلاینتها به تابع نسل دوم کنید.
- با محرکهای پسزمینه، هر دو نسل اول و دوم توابع بلافاصله پس از استقرار به هر رویدادی پاسخ میدهند.
- وقتی تمام ترافیک منتقل شد، تابع نسل اول را با استفاده از دستور firebase
firebase functions:delete
حذف کنید.- به صورت اختیاری، نام تابع نسل دوم را تغییر دهید تا با نام تابع نسل اول مطابقت داشته باشد.