Questo documento mostra come utilizzare l'autenticazione Firebase per accedere agli utenti a un'estensione Chrome che utilizza Manifest V3 .
Firebase Authentication fornisce più metodi di autenticazione per accedere agli utenti da un'estensione di Chrome, alcuni richiedono uno sforzo di sviluppo maggiore rispetto ad altri.
Per utilizzare i seguenti metodi in un'estensione Chrome Manifest V3, devi solo importarli da firebase/auth/web-extension
:
- Accedi con email e password (
createUserWithEmailAndPassword
esignInWithEmailAndPassword
) - Accedi con il collegamento e-mail (
sendSignInLinkToEmail
,isSignInWithEmailLink
esignInWithEmailLink
) - Accedi in modo anonimo (
signInAnonymously
) - Accedi con un sistema di autenticazione personalizzato (
signInWithCustomToken
) - Gestisci l'accesso del provider in modo indipendente, quindi utilizza
signInWithCredential
Sono supportati anche i seguenti metodi di accesso, ma richiedono alcune operazioni aggiuntive:
- Accedi con una finestra popup (
signInWithPopup
,linkWithPopup
ereauthenticateWithPopup
) - Accedi reindirizzando alla pagina di accesso (
signInWithRedirect
,linkWithRedirect
ereauthenticateWithRedirect
) - Accedi con il numero di telefono con reCAPTCHA
- Autenticazione SMS a più fattori con reCAPTCHA
- reCAPTCHA Protezione aziendale
Per utilizzare questi metodi in un'estensione Chrome Manifest V3, devi utilizzare Offscreen Documents .
Utilizza il punto di ingresso firebase/auth/web-extension
L'importazione da firebase/auth/web-extension
rende l'accesso degli utenti da un'estensione di Chrome simile a un'app Web.
firebase/auth/web-extension è supportato solo nelle versioni Web SDK v10.8.0 e successive.
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; });
Utilizza documenti fuori schermo
Alcuni metodi di autenticazione, come signInWithPopup
, linkWithPopup
e reauthenticateWithPopup
, non sono direttamente compatibili con le estensioni di Chrome perché richiedono il caricamento del codice dall'esterno del pacchetto di estensioni. A partire da Manifest V3, ciò non è consentito e verrà bloccato dalla piattaforma di estensione. Per aggirare questo problema, puoi caricare il codice all'interno di un iframe utilizzando un documento fuori schermo . Nel documento fuori schermo, implementare il normale flusso di autenticazione e inoltrare il risultato dal documento fuori schermo all'estensione.
Questa guida utilizza signInWithPopup
come esempio, ma lo stesso concetto si applica ad altri metodi di autenticazione.
Prima di iniziare
Questa tecnica richiede la configurazione di una pagina Web disponibile sul Web, che caricherai in un iframe. Qualsiasi host funziona per questo, incluso Firebase Hosting . Creare un sito web con il seguente contenuto:
<!DOCTYPE html> <html> <head> <title>signInWithPopup</title> <script src="signInWithPopup.js"></script> </head> <body><h1>signInWithPopup</h1></body> </html>
Accesso federato
Se utilizzi l'accesso federato, ad esempio l'accesso con Google, Apple, SAML o OIDC, devi aggiungere l'ID dell'estensione Chrome all'elenco dei domini autorizzati:
- Apri il tuo progetto nella console Firebase .
- Nella sezione Autenticazione , apri la pagina Impostazioni .
- Aggiungi un URI come il seguente all'elenco dei domini autorizzati:
chrome-extension://CHROME_EXTENSION_ID
Nel file manifest dell'estensione Chrome assicurati di aggiungere i seguenti URL alla lista consentita content_security_policy
:
-
https://apis.google.com
-
https://www.gstatic.com
-
https://www.googleapis.com
-
https://securetoken.googleapis.com
Implementare l'autenticazione
Nel tuo documento HTML, signInWithPopup.js è il codice JavaScript che gestisce l'autenticazione. Esistono due modi diversi per implementare un metodo supportato direttamente nell'estensione:
- Utilizzare
firebase/auth
anzichéfirebase/auth/web-extension
. Il punto di ingressoweb-extension
è per il codice in esecuzione all'interno dell'estensione. Anche se questo codice alla fine viene eseguito nell'estensione (in un iframe, nel documento fuori schermo), il contesto in cui viene eseguito è il web. - Avvolgere la logica di autenticazione in un ascoltatore
postMessage
, per proxyare la richiesta e la risposta di autenticazione.
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) } });
Crea la tua estensione Chrome
Una volta che il tuo sito web sarà attivo, potrai utilizzarlo nella tua estensione di Chrome.
- Aggiungi l'autorizzazione
offscreen
al tuo file manifest.json: - Crea il documento fuori schermo stesso. Questo è un file HTML minimo all'interno del tuo pacchetto di estensione che carica la logica del tuo documento JavaScript fuori schermo:
- Includi
offscreen.js
nel tuo pacchetto di estensioni. Funziona come proxy tra il sito web pubblico configurato nel passaggio 1 e la tua estensione. - Configura il documento fuori schermo dal tuo operatore del servizio 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; }
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; }
Ora, quando chiami firebaseAuth()
all'interno del tuo service operator, creerà il documento fuori schermo e caricherà il sito in un iframe. L'iframe verrà elaborato in background e Firebase seguirà il flusso di autenticazione standard. Una volta risolto o rifiutato, l'oggetto di autenticazione verrà inviato tramite proxy dal tuo iframe al tuo addetto al servizio, utilizzando il documento fuori schermo.