يوضّح هذا المستند كيفية استخدام Firebase Authentication لتسجيل دخول المستخدمين إلى إضافة Chrome تستخدم الإصدار Manifest V3.
Firebase Authentication توفّر طرق مصادقة متعدّدة لتسجيل دخول المستخدمين من إضافة Chrome، ويتطلّب بعضها جهدًا أكبر في التطوير مقارنةً بالبعض الآخر.
لاستخدام الطرق التالية في إضافة Chrome تستخدم الإصدار Manifest V3، ما عليك سوى
استيرادها من firebase/auth/web-extension:
- تسجيل الدخول باستخدام عنوان البريد الإلكتروني وكلمة المرور (
createUserWithEmailAndPasswordوsignInWithEmailAndPassword) - تسجيل الدخول باستخدام رابط البريد الإلكتروني (
sendSignInLinkToEmailوisSignInWithEmailLinkوsignInWithEmailLink) - تسجيل الدخول بشكل مجهول (
signInAnonymously) - تسجيل الدخول باستخدام نظام مصادقة مخصّص (
signInWithCustomToken) - التعامل مع تسجيل الدخول إلى موفّر الخدمة بشكل مستقل ثم استخدام
signInWithCredential
تتوفّر أيضًا طرق تسجيل الدخول التالية، ولكنها تتطلّب بعض العمل الإضافي:
- تسجيل الدخول باستخدام نافذة منبثقة (
signInWithPopupوlinkWithPopupوreauthenticateWithPopup) - تسجيل الدخول عن طريق إعادة التوجيه إلى صفحة تسجيل الدخول (
signInWithRedirectوlinkWithRedirectوreauthenticateWithRedirect) - تسجيل الدخول باستخدام رقم الهاتف مع reCAPTCHA
- المصادقة المتعدّدة العوامل عبر الرسائل القصيرة باستخدام reCAPTCHA
- الحماية باستخدام reCAPTCHA Enterprise
لاستخدام هذه الطرق في إضافة Chrome تستخدم الإصدار Manifest V3، عليك استخدام المستندات خارج الشاشة.
استخدام نقطة الدخول firebase/auth/web-extension
يؤدي الاستيراد من firebase/auth/web-extension إلى جعل تسجيل دخول المستخدمين من إضافة Chrome مشابهًا لتسجيل الدخول من تطبيق ويب.
لا تتوافق نقطة الدخول firebase/auth/web-extension إلا مع الإصدارات v10.8.0 والإصدارات الأحدث من حزمة تطوير البرامج (SDK) على الويب.
import { getAuth, signInWithEmailAndPassword } from 'firebase/auth/web-extension'; const auth = getAuth(); signInWithEmailAndPassword(auth, email, password) .then((userCredential) => { // Signed in const user = userCredential.user; // ... }) .catch((error) => { const errorCode = error.code; const errorMessage = error.message; });
استخدام المستندات خارج الشاشة
لا تتوافق بعض طرق المصادقة، مثل signInWithPopup وlinkWithPopup وreauthenticateWithPopup، بشكل مباشر مع إضافات Chrome، لأنّها تتطلّب تحميل الرمز من خارج حزمة الإضافة.
اعتبارًا من الإصدار Manifest V3، لا يُسمح بذلك وسيتم حظره من قِبل منصة الإضافة. لتجنُّب ذلك، يمكنك تحميل هذا الرمز ضِمن
إطار iframe باستخدام مستند خارج الشاشة.
في المستند خارج الشاشة، نفِّذ عملية المصادقة العادية وأعِد توجيه النتيجة من المستند خارج الشاشة إلى الإضافة.
يستخدم هذا الدليل signInWithPopup كمثال، ولكن ينطبق المفهوم نفسه على طرق المصادقة الأخرى.
قبل البدء
تتطلّب هذه الطريقة إعداد صفحة ويب متاحة على الويب، والتي ستحمّلها في إطار iframe. يمكنك استخدام أي مضيف لذلك، بما في ذلك استضافة Firebase. أنشِئ موقعًا إلكترونيًا يتضمّن المحتوى التالي:
<!DOCTYPE html>
<html>
<head>
<title>signInWithPopup</title>
<script src="signInWithPopup.js"></script>
</head>
<body><h1>signInWithPopup</h1></body>
</html>تسجيل الدخول الموحّد
إذا كنت تستخدم تسجيل الدخول الموحّد، مثل تسجيل الدخول باستخدام حساب Google أو Apple أو SAML أو OIDC، عليك إضافة رقم تعريف إضافة Chrome إلى قائمة النطاقات المفوّضة:
في وحدة تحكّم Firebase، انتقِل إلى الأمان > المصادقة.
في علامة التبويب الإعدادات ، انقر على إضافة نطاق في قسم النطاقات المفوّضة ، ثم أضِف معرّف موارد موحّد (URI) على النحو التالي:
chrome-extension://CHROME_EXTENSION_ID
في ملف بيان إضافة Chrome، تأكَّد من إضافة عناوين URL التالية إلى القائمة المسموح بها content_security_policy:
https://apis.google.comhttps://www.gstatic.comhttps://www.googleapis.comhttps://securetoken.googleapis.com
تنفيذ المصادقة
في مستند HTML، يكون signInWithPopup.js هو رمز JavaScript الذي يتعامل مع المصادقة. هناك طريقتان مختلفتان لتنفيذ طريقة متوافقة مباشرةً مع الإضافة:
- استخدِم
firebase/auth/web-extensionفي رمز الإضافة، مثل النصوص البرمجية للخلفية أو عاملي الخدمة أو النصوص البرمجية للنوافذ المنبثقة. استخدِمfirebase/authداخل إطار iframe خارج الشاشة فقط، لأنّ هذا الإطار يعمل في سياق صفحة ويب عادية. - غلِّف منطق المصادقة في مستمع
postMessage، وذلك لإعادة توجيه طلب المصادقة والردّ.
import { signInWithPopup, GoogleAuthProvider, getAuth } from'firebase/auth'; import { initializeApp } from 'firebase/app'; import firebaseConfig from './firebaseConfig.js' const app = initializeApp(firebaseConfig); const auth = getAuth(); // This code runs inside of an iframe in the extension's offscreen document. // This gives you a reference to the parent frame, i.e. the offscreen document. // You will need this to assign the targetOrigin for postMessage. const PARENT_FRAME = document.location.ancestorOrigins[0]; // This demo uses the Google auth provider, but any supported provider works. // Make sure that you enable any provider you want to use in the Firebase Console. // https://console.firebase.google.com/project/_/authentication/providers const PROVIDER = new GoogleAuthProvider(); function sendResponse(result) { globalThis.parent.self.postMessage(JSON.stringify(result), PARENT_FRAME); } globalThis.addEventListener('message', function({data}) { if (data.initAuth) { // Opens the Google sign-in page in a popup, inside of an iframe in the // extension's offscreen document. // To centralize logic, all respones are forwarded to the parent frame, // which goes on to forward them to the extension's service worker. signInWithPopup(auth, PROVIDER) .then(sendResponse) .catch(sendResponse) } });
إنشاء إضافة Chrome
بعد أن يصبح موقعك الإلكتروني متاحًا، يمكنك استخدامه في إضافة Chrome.
- أضِف إذن
offscreenإلى ملف manifest.json: - أنشِئ المستند خارج الشاشة نفسه. هذا هو ملف HTML بسيط ضِمن حزمة الإضافة يحمِّل منطق JavaScript للمستند خارج الشاشة:
- أدرِج
offscreen.jsفي حزمة الإضافة. يعمل هذا الملف كوكيل بين الـ موقع الإلكتروني العام الذي تم إعداده في الخطوة 1 والإضافة. - يمكنك إعداد المستند خارج الشاشة من عامل الخدمة background.js.
{ "name": "signInWithPopup Demo", "manifest_version" 3, "background": { "service_worker": "background.js" }, "permissions": [ "offscreen" ] }
<!DOCTYPE html>
<script src="./offscreen.js"></script>
// This URL must point to the public site const _URL = 'https://example.com/signInWithPopupExample'; const iframe = document.createElement('iframe'); iframe.src = _URL; document.documentElement.appendChild(iframe); chrome.runtime.onMessage.addListener(handleChromeMessages); function handleChromeMessages(message, sender, sendResponse) { // Extensions may have an number of other reasons to send messages, so you // should filter out any that are not meant for the offscreen document. if (message.target !== 'offscreen') { return false; } function handleIframeMessage({data}) { try { if (data.startsWith('!_{')) { // Other parts of the Firebase library send messages using postMessage. // You don't care about them in this context, so return early. return; } data = JSON.parse(data); self.removeEventListener('message', handleIframeMessage); sendResponse(data); } catch (e) { console.log(`json parse failed - ${e.message}`); } } globalThis.addEventListener('message', handleIframeMessage, false); // Initialize the authentication flow in the iframed document. You must set the // second argument (targetOrigin) of the message in order for it to be successfully // delivered. iframe.contentWindow.postMessage({"initAuth": true}, new URL(_URL).origin); return true; }
import { getAuth } from 'firebase/auth/web-extension'; const OFFSCREEN_DOCUMENT_PATH = '/offscreen.html'; // A global promise to avoid concurrency issues let creatingOffscreenDocument; // Chrome only allows for a single offscreenDocument. This is a helper function // that returns a boolean indicating if a document is already active. async function hasDocument() { // Check all windows controlled by the service worker to see if one // of them is the offscreen document with the given path const matchedClients = await clients.matchAll(); return matchedClients.some( (c) => c.url === chrome.runtime.getURL(OFFSCREEN_DOCUMENT_PATH) ); } async function setupOffscreenDocument(path) { // If we do not have a document, we are already setup and can skip if (!(await hasDocument())) { // create offscreen document if (creating) { await creating; } else { creating = chrome.offscreen.createDocument({ url: path, reasons: [ chrome.offscreen.Reason.DOM_SCRAPING ], justification: 'authentication' }); await creating; creating = null; } } } async function closeOffscreenDocument() { if (!(await hasDocument())) { return; } await chrome.offscreen.closeDocument(); } function getAuth() { return new Promise(async (resolve, reject) => { const auth = await chrome.runtime.sendMessage({ type: 'firebase-auth', target: 'offscreen' }); auth?.name !== 'FirebaseError' ? resolve(auth) : reject(auth); }) } async function firebaseAuth() { await setupOffscreenDocument(OFFSCREEN_DOCUMENT_PATH); const auth = await getAuth() .then((auth) => { console.log('User Authenticated', auth); return auth; }) .catch(err => { if (err.code === 'auth/operation-not-allowed') { console.error('You must enable an OAuth provider in the Firebase' + ' console in order to use signInWithPopup. This sample' + ' uses Google by default.'); } else { console.error(err); return err; } }) .finally(closeOffscreenDocument) return auth; }
الآن، عند استدعاء firebaseAuth() ضِمن عامل الخدمة، سيتم إنشاء المستند خارج الشاشة وتحميل الموقع الإلكتروني في إطار iframe. ستتم معالجة إطار iframe هذا في الخلفية، وستتّبع Firebase عملية المصادقة العادية. بعد أن يتم حلّ كائن المصادقة أو رفضه، سيتم إعادة توجيهه من إطار iframe إلى عامل الخدمة باستخدام المستند خارج الشاشة.