/*
# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
*/
/*
original:
http://github.com/piroor/webextensions-lib-tab-favicon-helper
*/
'use strict';
const TabFavIconHelper = {
LAST_EFFECTIVE_FAVICON: 'last-effective-favIcon',
VALID_FAVICON_PATTERN: /^(about|app|chrome|data|file|ftp|https?|moz-extension|resource):/,
DRAWABLE_FAVICON_PATTERN: /^(https?|moz-extension|resource):/,
// original: chrome://browser/content/aboutlogins/icons/favicon.svg
FAVICON_LOCKWISE: `
`,
// original: chrome://browser/content/robot.ico
FAVICON_ROBOT: `

`.trim(),
// original: chrome://browser/skin/controlcenter/dashboard.svg
FAVICON_DASHBOARD: `
`,
// original: chrome://browser/skin/developer.svg
FAVICON_DEVELOPER: `
`,
// original: chrome://browser/skin/privatebrowsing/favicon.svg
FAVICON_PRIVATE_BROWSING: `
`,
// original: chrome://browser/skin/settings.svg
FAVICON_SETTINGS: `
`,
// original: chrome://browser/skin/window.svg
FAVICON_WINDOW: `
`,
// original: chrome://devtools/skin/images/profiler-stopwatch.svg
FAVICON_PROFILER: `
`,
// original: chrome://global/skin/icons/performance.svg
FAVICON_PERFORMANCE: `
`,
// original: chrome://global/skin/icons/warning.svg
FAVICON_WARNING: `
`,
// original: chrome://mozapps/skin/extensions/extensionGeneric-16.svg
FAVICON_EXTENSION: `
`,
// original: globe-16.svg
FAVICON_GLOBE: `
`,
async _urlToKey(url) { // sha1 hash
const encoder = new TextEncoder();
const data = encoder.encode(url);
const hashBuffer = await crypto.subtle.digest('SHA-1', data);
const hashArray = Array.from(new Uint8Array(hashBuffer));
const hashHex = hashArray.map((byte) => byte.toString(16).padStart(2, '0')).join('');
return hashHex;
},
DB_NAME: 'TabFavIconHelper',
DB_VERSION: 2,
STORE_FAVICONS: 'favIcons',
STORE_EFFECTIVE_FAVICONS: 'effectiveFavIcons',
STORE_UNEFFECTIVE_FAVICONS: 'uneffectiveFavIcons',
EXPIRATION_TIME_IN_MSEC: 7 * 24 * 60 * 60 * 1000, // 7 days
async _openDB() {
if (this._openedDB)
return this._openedDB;
return new Promise((resolve, _reject) => {
const request = indexedDB.open(this.DB_NAME, this.DB_VERSION);
request.onerror = () => {
// This can fail if this is in a private window.
// See: https://github.com/piroor/treestyletab/issues/3387
//reject(new Error('Failed to open database'));
resolve(null);
};
request.onsuccess = () => {
const db = request.result;
this._openedDB = db;
resolve(db);
};
request.onupgradeneeded = (event) => {
const db = event.target.result;
const objectStores = db.objectStoreNames;
const needToUpgrade = event.oldVersion < this.DB_VERSION;
if (needToUpgrade) {
if (objectStores.contains(this.STORE_FAVICONS))
db.deleteObjectStore(this.STORE_FAVICONS);
if (objectStores.contains(this.STORE_EFFECTIVE_FAVICONS))
db.deleteObjectStore(this.STORE_EFFECTIVE_FAVICONS);
if (objectStores.contains(this.STORE_UNEFFECTIVE_FAVICONS))
db.deleteObjectStore(this.STORE_UNEFFECTIVE_FAVICONS);
}
if (needToUpgrade ||
!objectStores.contains(this.STORE_FAVICONS)) {
const favIconsStore = db.createObjectStore(this.STORE_FAVICONS, { keyPath: 'key', unique: true });
favIconsStore.createIndex('urlKey', 'urlKey', { unique: false });
favIconsStore.createIndex('timestamp', 'timestamp');
}
if (needToUpgrade ||
!objectStores.contains(this.STORE_EFFECTIVE_FAVICONS)) {
const effectiveFavIconsStore = db.createObjectStore(this.STORE_EFFECTIVE_FAVICONS, { keyPath: 'urlKey', unique: true });
effectiveFavIconsStore.createIndex('timestamp', 'timestamp');
effectiveFavIconsStore.createIndex('favIconKey', 'favIconKey', { unique: false });
}
if (needToUpgrade ||
!objectStores.contains(this.STORE_UNEFFECTIVE_FAVICONS)) {
const uneffectiveFavIconsStore = db.createObjectStore(this.STORE_UNEFFECTIVE_FAVICONS, { keyPath: 'urlKey', unique: true });
uneffectiveFavIconsStore.createIndex('timestamp', 'timestamp');
uneffectiveFavIconsStore.createIndex('favIconKey', 'favIconKey', { unique: false });
}
};
});
},
async _associateFavIconUrlToTabUrl({ favIconUrl, tabUrl, store } = {}) {
const [db, tabUrlKey, favIconKey] = await Promise.all([
this._openDB(),
this._urlToKey(tabUrl),
this._urlToKey(favIconUrl),
]);
if (!db)
return;
try {
const transaction = db.transaction([store, this.STORE_FAVICONS], 'readwrite');
const associationStore = transaction.objectStore(store);
const favIconStore = transaction.objectStore(this.STORE_FAVICONS);
const timestamp = Date.now();
const associationRequest = associationStore.put({ urlKey: tabUrlKey, favIconKey, timestamp });
const favIconRequest = favIconStore.put({ key: favIconKey, url: favIconUrl, timestamp });
transaction.oncomplete = () => {
//db.close();
this._reserveToExpireOldEntries();
favIconUrl = undefined;
tabUrl = undefined;
store = undefined;
};
associationRequest.onerror = event => {
console.error(`Failed to associate favIconUrl ${favIconUrl} to tabUrl ${tabUrl} in the store ${store}`, event);
};
favIconRequest.onerror = event => {
console.error(`Failed to store favIconUrl ${favIconUrl} to tabUrl ${tabUrl} in the store ${store}`, event);
};
}
catch(error) {
console.error(`Failed to associate favIconUrl ${favIconUrl} to tabUrl ${tabUrl} in the store ${store}`, error);
}
},
async _unassociateFavIconUrlFromTabUrl({ tabUrl, store } = {}) {
const [db, tabUrlKey] = await Promise.all([
this._openDB(),
this._urlToKey(tabUrl),
]);
if (!db)
return;
try {
const transaction = db.transaction([store], 'readwrite');
const associationStore = transaction.objectStore(store);
const unassociationRequest = associationStore.delete(tabUrlKey);
transaction.oncomplete = () => {
//db.close();
this._reserveToExpireOldEntries();
tabUrl = undefined;
store = undefined;
};
unassociationRequest.onerror = event => {
console.error(`Failed to unassociate favIconUrl from tabUrl ${tabUrl} in the store ${store}`, event);
};
}
catch(error) {
console.error(`Failed to unassociate favIconUrl from tabUrl ${tabUrl} in the store ${store}`, error);
}
},
async _getAssociatedFavIconUrlFromTabUrl({ tabUrl, store } = {}) {
return new Promise(async (resolve, _reject) => {
const [db, tabUrlKey] = await Promise.all([
this._openDB(),
this._urlToKey(tabUrl),
]);
if (!db) {
resolve(null);
return;
}
try {
const transaction = db.transaction([store, this.STORE_FAVICONS], 'readonly');
const associationStore = transaction.objectStore(store);
const favIconStore = transaction.objectStore(this.STORE_FAVICONS);
const associationRequest = associationStore.get(tabUrlKey);
associationRequest.onsuccess = () => {
const association = associationRequest.result;
if (!association) {
//console.log(`No associated favIconUrl for the tabUrl ${tabUrl} in the store ${store}`);
resolve(null);
return;
}
const favIconRequest = favIconStore.get(association.favIconKey);
favIconRequest.onsuccess = () => {
let favIcon = favIconRequest.result;
if (!favIcon) {
//console.log(`FavIcon data not found for the tabUrl ${tabUrl} in the store ${store}`);
resolve(null);
return;
}
if (favIcon.timestamp < Date.now() - this.EXPIRATION_TIME_IN_MSEC) {
//console.log(`FavIcon data is expired for the tabUrl ${tabUrl} in the store ${store}`);
this._reserveToExpireOldEntries();
resolve(null);
return;
}
resolve(favIcon.url);
favIcon.url = undefined;
favIcon = undefined;
};
favIconRequest.onerror = event => {
console.error(`Failed to get favIconUrl from tabUrl ${tabUrl}`, event);
resolve(null);
};
};
associationRequest.onerror = event => {
console.error(`Failed to get favIcon association from tabUrl ${tabUrl}`, event);
resolve(null);
};
transaction.oncomplete = () => {
//db.close();
tabUrl = undefined;
store = undefined;
};
}
catch(error) {
console.error('Failed to get from cache:', error);
resolve(null);
}
});
},
async _reserveToExpireOldEntries() {
if (this._reservedExpiration)
clearTimeout(this._reservedExpiration);
this._reservedExpiration = setTimeout(() => {
this._reservedExpiration = null;
this._expireOldEntries();
}, 500);
},
async _expireOldEntries() {
return new Promise(async (resolve, reject) => {
const db = await this._openDB();
if (!db) {
resolve();
return;
}
try {
const transaction = db.transaction([this.STORE_FAVICONS, this.STORE_EFFECTIVE_FAVICONS, this.STORE_UNEFFECTIVE_FAVICONS], 'readwrite');
const favIconsStore = transaction.objectStore(this.STORE_FAVICONS);
const effectiveFavIconsStore = transaction.objectStore(this.STORE_EFFECTIVE_FAVICONS);
const uneffectiveFavIconsStore = transaction.objectStore(this.STORE_UNEFFECTIVE_FAVICONS);
const favIconIndex = favIconsStore.index('timestamp');
const effectiveFavIconIndex = effectiveFavIconsStore.index('timestamp');
const uneffectiveFavIconIndex = uneffectiveFavIconsStore.index('timestamp');
const expirationTimestamp = Date.now() - this.EXPIRATION_TIME_IN_MSEC;
const favIconRequest = favIconIndex.openCursor(IDBKeyRange.upperBound(expirationTimestamp));
favIconRequest.onsuccess = (event) => {
const cursor = event.target.result;
if (!cursor)
return;
const key = cursor.primaryKey;
cursor.continue();
const deleteRequest = favIconsStore.delete(key);
deleteRequest.onerror = event => {
console.error(`Failed to clear favicon index`, event);
};
};
favIconRequest.onerror = event => {
console.error(`Failed to retrieve favicon index`, event);
};
const effectiveFavIconRequest = effectiveFavIconIndex.openCursor(IDBKeyRange.upperBound(expirationTimestamp));
effectiveFavIconRequest.onsuccess = (event) => {
const cursor = event.target.result;
if (!cursor)
return;
const url = cursor.primaryKey;
cursor.continue();
const deleteRequest = effectiveFavIconsStore.delete(url);
deleteRequest.onerror = event => {
console.error(`Failed to clear effective favicon index`, event);
};
};
effectiveFavIconRequest.onerror = event => {
console.error(`Failed to retrieve effective favicon index`, event);
};
const uneffectiveFavIconRequest = uneffectiveFavIconIndex.openCursor(IDBKeyRange.upperBound(expirationTimestamp));
uneffectiveFavIconRequest.onsuccess = (event) => {
const cursor = event.target.result;
if (!cursor)
return;
const url = cursor.primaryKey;
cursor.continue();
const deleteRequest = uneffectiveFavIconsStore.delete(url);
deleteRequest.onerror = event => {
console.error(`Failed to clear uneffective favicon index`, event);
};
};
uneffectiveFavIconRequest.onerror = event => {
console.error(`Failed to retrieve uneffective favicon index`, event);
};
transaction.oncomplete = () => {
//db.close();
resolve();
};
}
catch(error) {
console.error('Failed to expire old entries:', error);
reject(error);
}
});
},
_tasks: [],
_processStep: 5,
FAVICON_SIZE: 16,
_init() {
this._onTabUpdated = this._onTabUpdated.bind(this);
browser.tabs.onUpdated.addListener(this._onTabUpdated);
this.canvas = document.createElement('canvas');
this.canvas.width = this.canvas.height = this.FAVICON_SIZE;
this.canvas.setAttribute('style', `
visibility: hidden;
pointer-events: none;
position: fixed
`);
document.body.appendChild(this.canvas);
window.addEventListener('unload', () => {
browser.tabs.onUpdated.removeListener(this._onTabUpdated);
}, { once: true });
},
_sessionAPIAvailable: (
browser.sessions &&
browser.sessions.getTabValue &&
browser.sessions.setTabValue &&
browser.sessions.removeTabValue
),
_addTask(task) {
this._tasks.push(task);
this._run();
},
_run() {
if (this._running)
return;
this._running = true;
const processOneTask = () => {
if (this._tasks.length == 0) {
this._running = false;
}
else {
const tasks = this._tasks.splice(0, this._processStep);
while (tasks.length > 0) {
tasks.shift()();
}
window.requestAnimationFrame(processOneTask);
}
};
processOneTask();
},
// public
loadToImage(params = {}) {
this._addTask(() => {
this._getEffectiveFavIconURL(params.tab, params.url)
.then(url => {
params.image.src = url;
params.image.classList.remove('error');
url = undefined;
},
_error => {
params.image.src = '';
params.image.classList.add('error');
});
});
},
// public
maybeImageTab(_tab) { // for backward compatibility
return false;
},
_getSafeFaviconUrl(url) {
switch (url) {
case 'chrome://browser/content/aboutlogins/icons/favicon.svg':
return this._getSVGDataURI(this.FAVICON_LOCKWISE);
case 'chrome://browser/content/robot.ico':
return this.FAVICON_ROBOT;
case 'chrome://browser/skin/controlcenter/dashboard.svg':
return this._getSVGDataURI(this.FAVICON_DASHBOARD);
case 'chrome://browser/skin/developer.svg':
return this._getSVGDataURI(this.FAVICON_DEVELOPER);
case 'chrome://browser/skin/privatebrowsing/favicon.svg':
return this._getSVGDataURI(this.FAVICON_PRIVATE_BROWSING);
case 'chrome://browser/skin/settings.svg':
return this._getSVGDataURI(this.FAVICON_SETTINGS);
case 'chrome://browser/skin/window.svg':
return this._getSVGDataURI(this.FAVICON_WINDOW);
case 'chrome://devtools/skin/images/profiler-stopwatch.svg':
return this._getSVGDataURI(this.FAVICON_PROFILER);
case 'chrome://global/skin/icons/performance.svg':
return this._getSVGDataURI(this.FAVICON_PERFORMANCE);
case 'chrome://global/skin/icons/warning.svg':
return this._getSVGDataURI(this.FAVICON_WARNING);
case 'chrome://mozapps/skin/extensions/extensionGeneric-16.svg':
return this._getSVGDataURI(this.FAVICON_EXTENSION);
default:
if (/^chrome:\/\//.test(url) &&
!/^chrome:\/\/branding\//.test(url))
return this._getSVGDataURI(this.FAVICON_GLOBE);
break;
}
return url;
},
_getSVGDataURI(svg) {
return `data:image/svg+xml,${encodeURIComponent(svg.trim())}`;
},
// public
async getLastEffectiveFavIconURL(tab) {
if (tab.favIconUrl?.startsWith('data:'))
return tab.favIconUrl;
const uneffectiveFavIconUrl = await this._getAssociatedFavIconUrlFromTabUrl({ tabUrl: tab.url, store: this.STORE_UNEFFECTIVE_FAVICONS });
if (uneffectiveFavIconUrl)
return null;
const favIconUrl = await this._getAssociatedFavIconUrlFromTabUrl({ tabUrl: tab.url, store: this.STORE_EFFECTIVE_FAVICONS });
if (favIconUrl)
return favIconUrl;
if (!this._sessionAPIAvailable)
return null;
const lastData = await browser.sessions.getTabValue(tab.id, this.LAST_EFFECTIVE_FAVICON);
return lastData && lastData.url == tab.url && lastData.favIconUrl;
},
async _getEffectiveFavIconURL(tab, favIconUrl = null) {
if (tab.favIconUrl?.startsWith('data:')) {
browser.sessions.removeTabValue(tab.id, this.LAST_EFFECTIVE_FAVICON);
this._unassociateFavIconUrlFromTabUrl({ tabUrl: tab.url, store: this.STORE_UNEFFECTIVE_FAVICONS });
return tab.favIconUrl;
}
return new Promise(async (resolve, reject) => {
favIconUrl = this._getSafeFaviconUrl(favIconUrl || tab.favIconUrl);
let storedFavIconUrl;
if (!favIconUrl && tab.discarded) {
// discarded tab doesn't have favIconUrl, so we should use cached data.
storedFavIconUrl = favIconUrl = await this.getLastEffectiveFavIconURL(tab);
}
let loader, onLoad, onError;
const clear = (() => {
if (loader) {
loader.removeEventListener('load', onLoad, { once: true });
loader.removeEventListener('error', onError, { once: true });
}
loader = onLoad = onError = favIconUrl = storedFavIconUrl = undefined;
});
onLoad = async foundFavIconUrl => {
let dataURL = null;
if (this.DRAWABLE_FAVICON_PATTERN.test(favIconUrl)) {
const context = this.canvas.getContext('2d');
context.clearRect(0, 0, this.FAVICON_SIZE, this.FAVICON_SIZE);
context.drawImage(loader, 0, 0, this.FAVICON_SIZE, this.FAVICON_SIZE);
try {
dataURL = this.canvas.toDataURL('image/png');
}
catch(_error) {
// it can fail due to security reasons
}
}
const oldFavIconUrl = foundFavIconUrl || await this._getAssociatedFavIconUrlFromTabUrl({ tabUrl: tab.url, store: this.STORE_EFFECTIVE_FAVICONS });
if (!oldFavIconUrl ||
oldFavIconUrl != favIconUrl) {
if (this._sessionAPIAvailable)
browser.sessions.setTabValue(tab.id, this.LAST_EFFECTIVE_FAVICON, {
url: tab.url,
favIconUrl,
});
}
this._associateFavIconUrlToTabUrl({ tabUrl: tab.url, favIconUrl, store: this.STORE_EFFECTIVE_FAVICONS });
this._unassociateFavIconUrlFromTabUrl({ tabUrl: tab.url, store: this.STORE_UNEFFECTIVE_FAVICONS });
resolve(dataURL || favIconUrl);
clear();
};
onError = async error => {
this._unassociateFavIconUrlFromTabUrl({ tabUrl: tab.url, store: this.STORE_EFFECTIVE_FAVICONS });
this._associateFavIconUrlToTabUrl({ tabUrl: tab.url, favIconUrl, store: this.STORE_UNEFFECTIVE_FAVICONS });
if (this._sessionAPIAvailable)
browser.sessions.removeTabValue(tab.id, this.LAST_EFFECTIVE_FAVICON);
clear();
reject(error || new Error('No effective icon'));
};
storedFavIconUrl = storedFavIconUrl || await this._getAssociatedFavIconUrlFromTabUrl({ tabUrl: tab.url, store: this.STORE_EFFECTIVE_FAVICONS });
if (storedFavIconUrl)
return onLoad(storedFavIconUrl);
if (!favIconUrl ||
!this.VALID_FAVICON_PATTERN.test(favIconUrl)) {
onError();
return;
}
loader = new Image();
if (/^https?:/.test(favIconUrl))
loader.crossOrigin = 'anonymous';
loader.addEventListener('load', () => onLoad(), { once: true });
loader.addEventListener('error', onError, { once: true });
try {
loader.src = favIconUrl;
}
catch(error) {
this._unassociateFavIconUrlFromTabUrl({ tabUrl: tab.url, store: this.STORE_EFFECTIVE_FAVICONS });
this._associateFavIconUrlToTabUrl({ tabUrl: tab.url, favIconUrl, store: this.STORE_UNEFFECTIVE_FAVICONS });
onError(error);
}
});
},
_onTabUpdated(tabId, changeInfo, _tab) {
if (!this._hasFavIconInfo(changeInfo))
return;
let timer = this._updatingTabs.get(tabId);
if (timer)
clearTimeout(timer);
// Updating of last effective favicon must be done after the loading
// of the tab itself is correctly done, to avoid cookie problems on
// some websites.
// See also: https://github.com/piroor/treestyletab/issues/2064
timer = setTimeout(async () => {
this._updatingTabs.delete(tabId);
const tab = await browser.tabs.get(tabId);
if (!tab ||
(changeInfo.favIconUrl &&
tab.favIconUrl != changeInfo.favIconUrl) ||
(changeInfo.url &&
tab.url != changeInfo.url) ||
!this._hasFavIconInfo(tab))
return; // expired
await this._getEffectiveFavIconURL(
tab,
changeInfo.favIconUrl
).catch(_error => {});
}, 5000);
this._updatingTabs.set(tabId, timer);
},
_hasFavIconInfo(tabOrChangeInfo) {
return 'favIconUrl' in tabOrChangeInfo;
},
_updatingTabs: new Map(),
};
TabFavIconHelper._init(); // eslint-disable-line no-underscore-dangle
export default TabFavIconHelper;