Разработка расширения для браузера Firefox (Firefox Add-on)
Firefox Add-ons основаны на WebExtensions API — той же спецификации, что и Chrome Extensions. Большинство Chrome-расширений можно перенести на Firefox с минимальными правками. Но есть различия в деталях: поддержка API, процедура подписи, движок расширений.
Отличия Firefox от Chrome
| Аспект | Firefox | Chrome |
|---|---|---|
| API namespace | browser.* (Promise) + chrome.* |
chrome.* (Callback) |
| Manifest | MV2 и MV3 (MV3 добавлен в Firefox 109+) | Только MV3 (MV2 устарел) |
| Background | Persistent background page (MV2) или SW (MV3) | Только Service Worker |
| Подпись | Обязательна через AMO | Не требуется |
browser_style |
Поддерживается | Не поддерживается |
Manifest V2 (Firefox — наиболее стабильный вариант)
{
"manifest_version": 2,
"name": "My Firefox Add-on",
"version": "1.0.0",
"description": "Описание дополнения",
"permissions": [
"storage",
"tabs",
"activeTab",
"https://*.example.com/*"
],
"background": {
"scripts": ["background.js"],
"persistent": false
},
"browser_action": {
"default_popup": "popup.html",
"default_icon": {
"48": "icons/icon48.png",
"96": "icons/icon96.png"
}
},
"content_scripts": [{
"matches": ["https://*.target-site.com/*"],
"js": ["content.js"]
}],
"options_ui": {
"page": "options.html",
"open_in_tab": false
}
}
Promise-based API
Firefox поддерживает нативные Promise в browser.*. Полифил webextension-polyfill делает код совместимым с Chrome:
import browser from 'webextension-polyfill';
// Firefox — нативные промисы
const tabs = await browser.tabs.query({ active: true, currentWindow: true });
const tab = tabs[0];
// Выполнить скрипт (MV2)
await browser.tabs.executeScript(tab.id, {
code: 'document.body.style.background = "yellow"',
});
// Хранилище
await browser.storage.local.set({ key: 'value' });
const result = await browser.storage.local.get('key');
Content Script: ограничения и обходы
В Firefox Content Scripts работают в изолированном мире (Xray wrapper) — они не имеют доступа к JavaScript-объектам страницы напрямую. Для взаимодействия со страницей нужен window.postMessage или CustomEvent:
// Content script -> Page script
window.postMessage({ source: 'myExtension', type: 'REQUEST_DATA' }, '*');
// Page script обработчик
window.addEventListener('message', (e) => {
if (e.data?.source === 'myExtension' && e.data?.type === 'REQUEST_DATA') {
window.postMessage({
source: 'myPage',
type: 'RESPONSE_DATA',
payload: window.__MY_APP_STATE__,
}, '*');
}
});
Подпись и публикация
Firefox требует подписи от Mozilla для любого расширения, распространяемого не через AMO:
# web-ext — официальный CLI от Mozilla
npm install -g web-ext
# Разработка с hot reload
web-ext run --source-dir ./dist --firefox-binary "/path/to/firefox"
# Сборка
web-ext build --source-dir ./dist --artifacts-dir ./artifacts
# Подпись (нужны AMO API ключи)
web-ext sign \
--source-dir ./dist \
--api-key $AMO_JWT_ISSUER \
--api-secret $AMO_JWT_SECRET
Для получения API ключей: addons.mozilla.org/developers/addon/api/key/.
Временная установка для тестирования
Без подписи расширение можно установить только через about:debugging:
- Открыть
about:debugging#/runtime/this-firefox - «Загрузить временное дополнение»
- Выбрать
manifest.json - Работает до перезапуска браузера
Для корпоративного деплоя без AMO — использовать policies.json в Firefox Enterprise.
Специфика Firefox: contextualIdentities (контейнеры)
Firefox поддерживает Containers — изолированные контексты с разными куки. Доступ через API:
const containers = await browser.contextualIdentities.query({});
// { cookieStoreId, name, color, icon }
// Открыть вкладку в конкретном контейнере
await browser.tabs.create({
url: 'https://example.com',
cookieStoreId: 'firefox-container-1',
});
Сроки
Firefox Add-on, адаптированный из Chrome-расширения, с подписью и публикацией на AMO — 2–3 рабочих дня дополнительно к готовому Chrome-расширению. Разработка нового дополнения с нуля — 4–8 дней в зависимости от функциональности.







