mirror of
https://github.com/bitwarden/browser.git
synced 2024-12-31 17:57:43 +01:00
674 lines
20 KiB
JavaScript
674 lines
20 KiB
JavaScript
var isBackground = true;
|
|
var loginsToAdd = [];
|
|
var i18nService = new i18nService();
|
|
var constantsService = new ConstantsService();
|
|
var utilsService = new UtilsService();
|
|
var cryptoService = new CryptoService(constantsService);
|
|
var tokenService = new TokenService();
|
|
var apiService = new ApiService(tokenService);
|
|
var userService = new UserService(tokenService, apiService, cryptoService);
|
|
var settingsService = new SettingsService(userService);
|
|
var loginService = new LoginService(cryptoService, userService, apiService, settingsService);
|
|
var folderService = new FolderService(cryptoService, userService, apiService);
|
|
var syncService = new SyncService(loginService, folderService, userService, apiService, settingsService);
|
|
var autofillService = new AutofillService();
|
|
var passwordGenerationService = new PasswordGenerationService();
|
|
var appIdService = new AppIdService();
|
|
|
|
chrome.commands.onCommand.addListener(function (command) {
|
|
if (command === 'generate_password') {
|
|
ga('send', {
|
|
hitType: 'event',
|
|
eventAction: 'Generated Password From Command'
|
|
});
|
|
passwordGenerationService.getOptions().then(function (options) {
|
|
var password = passwordGenerationService.generatePassword(options);
|
|
copyToClipboard(password);
|
|
});
|
|
}
|
|
});
|
|
|
|
var loadMenuRan = false,
|
|
loginToAutoFill = null,
|
|
pageDetailsToAutoFill = [],
|
|
autofillTimeout = null;
|
|
|
|
chrome.runtime.onMessage.addListener(function (msg, sender, sendResponse) {
|
|
if (msg.command === 'loggedOut' || msg.command === 'loggedIn' || msg.command === 'unlocked' || msg.command === 'locked') {
|
|
if (loadMenuRan) {
|
|
return;
|
|
}
|
|
loadMenuRan = true;
|
|
|
|
setIcon();
|
|
refreshBadgeAndMenu();
|
|
}
|
|
else if (msg.command === 'syncCompleted' && msg.successfully) {
|
|
if (loadMenuRan) {
|
|
return;
|
|
}
|
|
loadMenuRan = true;
|
|
|
|
setTimeout(refreshBadgeAndMenu, 2000);
|
|
}
|
|
else if (msg.command === 'bgOpenOverlayPopup') {
|
|
messageCurrentTab('openOverlayPopup', msg.data);
|
|
}
|
|
else if (msg.command === 'bgCloseOverlayPopup') {
|
|
messageCurrentTab('closeOverlayPopup');
|
|
}
|
|
else if (msg.command === 'bgOpenNotificationBar') {
|
|
messageTab(sender.tab.id, 'openNotificationBar', msg.data);
|
|
}
|
|
else if (msg.command === 'bgCloseNotificationBar') {
|
|
messageTab(sender.tab.id, 'closeNotificationBar');
|
|
}
|
|
else if (msg.command === 'bgCollectPageDetails') {
|
|
collectPageDetailsForContentScript(sender.tab);
|
|
}
|
|
else if (msg.command === 'bgAddLogin') {
|
|
addLogin(msg.login, sender.tab);
|
|
}
|
|
else if (msg.command === 'bgAddClose') {
|
|
removeAddLogin(sender.tab);
|
|
}
|
|
else if (msg.command === 'bgAddSave') {
|
|
saveAddLogin(sender.tab);
|
|
}
|
|
else if (msg.command === 'collectPageDetailsResponse') {
|
|
// messageCurrentTab('openNotificationBar', { type: 'add', typeData: null });
|
|
if (msg.contentScript) {
|
|
var forms = autofillService.getFormsWithPasswordFields(msg.details);
|
|
messageTab(msg.tabId, 'pageDetails', { details: msg.details, forms: forms });
|
|
}
|
|
else {
|
|
clearTimeout(autofillTimeout);
|
|
pageDetailsToAutoFill.push({ frameId: sender.frameId, tabId: msg.tabId, details: msg.details });
|
|
autofillTimeout = setTimeout(autofillPage, 300);
|
|
}
|
|
}
|
|
});
|
|
|
|
setIcon();
|
|
function setIcon() {
|
|
userService.isAuthenticated(function (isAuthenticated) {
|
|
cryptoService.getKey(false, function (key) {
|
|
var suffix = '';
|
|
if (!isAuthenticated) {
|
|
suffix = '_gray';
|
|
}
|
|
else if (!key) {
|
|
suffix = '_locked';
|
|
}
|
|
|
|
chrome.browserAction.setIcon({
|
|
path: {
|
|
'19': 'images/icon19' + suffix + '.png',
|
|
'38': 'images/icon38' + suffix + '.png',
|
|
}
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
if (chrome.runtime.onInstalled) {
|
|
chrome.runtime.onInstalled.addListener(function (details) {
|
|
ga('send', {
|
|
hitType: 'event',
|
|
eventAction: 'onInstalled ' + details.reason
|
|
});
|
|
|
|
if (details.reason === 'install') {
|
|
chrome.tabs.create({ url: 'https://bitwarden.com/browser-start/' }, function (tab) { });
|
|
}
|
|
});
|
|
}
|
|
|
|
function buildContextMenu(callback) {
|
|
chrome.contextMenus.removeAll(function () {
|
|
chrome.contextMenus.create({
|
|
type: 'normal',
|
|
id: 'root',
|
|
contexts: ['all'],
|
|
title: 'bitwarden'
|
|
}, function () {
|
|
chrome.contextMenus.create({
|
|
type: 'normal',
|
|
id: 'autofill',
|
|
parentId: 'root',
|
|
contexts: ['all'],
|
|
title: i18nService.autoFill
|
|
}, function () {
|
|
chrome.contextMenus.create({
|
|
type: 'normal',
|
|
id: 'copy-username',
|
|
parentId: 'root',
|
|
contexts: ['all'],
|
|
title: i18nService.copyUsername
|
|
}, function () {
|
|
chrome.contextMenus.create({
|
|
type: 'normal',
|
|
id: 'copy-password',
|
|
parentId: 'root',
|
|
contexts: ['all'],
|
|
title: i18nService.copyPassword
|
|
}, function () {
|
|
chrome.contextMenus.create({
|
|
type: 'separator',
|
|
parentId: 'root'
|
|
});
|
|
|
|
chrome.contextMenus.create({
|
|
type: 'normal',
|
|
id: 'generate-password',
|
|
parentId: 'root',
|
|
contexts: ['all'],
|
|
title: i18nService.generatePasswordCopied
|
|
}, function () {
|
|
if (callback) {
|
|
callback();
|
|
}
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
chrome.tabs.onActivated.addListener(function (activeInfo) {
|
|
checkLoginsToAdd();
|
|
refreshBadgeAndMenu();
|
|
});
|
|
|
|
var onReplacedRan = false;
|
|
chrome.tabs.onReplaced.addListener(function (addedTabId, removedTabId) {
|
|
if (onReplacedRan) {
|
|
return;
|
|
}
|
|
onReplacedRan = true;
|
|
checkLoginsToAdd();
|
|
refreshBadgeAndMenu();
|
|
});
|
|
|
|
var onUpdatedRan = false;
|
|
chrome.tabs.onUpdated.addListener(function (tabId, changeInfo, tab) {
|
|
if (onUpdatedRan) {
|
|
return;
|
|
}
|
|
onUpdatedRan = true;
|
|
checkLoginsToAdd();
|
|
refreshBadgeAndMenu();
|
|
});
|
|
|
|
function refreshBadgeAndMenu() {
|
|
chrome.tabs.query({ active: true }, function (tabs) {
|
|
var tab = null;
|
|
if (tabs.length > 0) {
|
|
tab = tabs[0];
|
|
}
|
|
|
|
if (!tab) {
|
|
return;
|
|
}
|
|
|
|
buildContextMenu(function () {
|
|
loadMenuAndUpdateBadge(tab.url, tab.id, true);
|
|
onUpdatedRan = onReplacedRan = loadMenuRan = false;
|
|
});
|
|
});
|
|
}
|
|
|
|
function loadMenuAndUpdateBadge(url, tabId, loadContextMenuOptions) {
|
|
if (!url) {
|
|
return;
|
|
}
|
|
|
|
var tabDomain = tldjs.getDomain(url);
|
|
if (!tabDomain) {
|
|
return;
|
|
}
|
|
|
|
chrome.browserAction.setBadgeBackgroundColor({ color: '#294e5f' });
|
|
|
|
loginService.getAllDecryptedForDomain(tabDomain).then(function (logins) {
|
|
sortLogins(logins);
|
|
for (var i = 0; i < logins.length; i++) {
|
|
if (loadContextMenuOptions) {
|
|
loadLoginContextMenuOptions(logins[i]);
|
|
}
|
|
}
|
|
|
|
if (logins.length > 0 && logins.length < 9) {
|
|
chrome.browserAction.setBadgeText({
|
|
text: logins.length.toString(),
|
|
tabId: tabId
|
|
});
|
|
}
|
|
else if (logins.length > 0) {
|
|
chrome.browserAction.setBadgeText({
|
|
text: '9+',
|
|
tabId: tabId
|
|
});
|
|
}
|
|
else {
|
|
loadNoLoginsContextMenuOptions(i18nService.noMatchingLogins);
|
|
chrome.browserAction.setBadgeText({
|
|
text: '',
|
|
tabId: tabId
|
|
});
|
|
}
|
|
}, function () {
|
|
loadNoLoginsContextMenuOptions(i18nService.vaultLocked);
|
|
chrome.browserAction.setBadgeText({
|
|
text: '',
|
|
tabId: tabId
|
|
});
|
|
});
|
|
}
|
|
|
|
chrome.contextMenus.onClicked.addListener(function (info, tab) {
|
|
if (info.menuItemId === 'generate-password') {
|
|
ga('send', {
|
|
hitType: 'event',
|
|
eventAction: 'Generated Password From Context Menu'
|
|
});
|
|
passwordGenerationService.getOptions().then(function (options) {
|
|
var password = passwordGenerationService.generatePassword(options);
|
|
copyToClipboard(password);
|
|
});
|
|
}
|
|
else if (info.parentMenuItemId === 'autofill' || info.parentMenuItemId === 'copy-username' ||
|
|
info.parentMenuItemId === 'copy-password') {
|
|
var id = info.menuItemId.split('_')[1];
|
|
if (id === 'noop') {
|
|
return;
|
|
}
|
|
|
|
loginService.getAllDecrypted().then(function (logins) {
|
|
for (var i = 0; i < logins.length; i++) {
|
|
if (logins[i].id === id) {
|
|
if (info.parentMenuItemId === 'autofill') {
|
|
ga('send', {
|
|
hitType: 'event',
|
|
eventAction: 'Autofilled From Context Menu'
|
|
});
|
|
startAutofillPage(logins[i]);
|
|
}
|
|
else if (info.parentMenuItemId === 'copy-username') {
|
|
ga('send', {
|
|
hitType: 'event',
|
|
eventAction: 'Copied Username From Context Menu'
|
|
});
|
|
copyToClipboard(logins[i].username);
|
|
}
|
|
else if (info.parentMenuItemId === 'copy-password') {
|
|
ga('send', {
|
|
hitType: 'event',
|
|
eventAction: 'Copied Password From Context Menu'
|
|
});
|
|
copyToClipboard(logins[i].password);
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
}, function () {
|
|
|
|
});
|
|
}
|
|
});
|
|
|
|
function messageCurrentTab(command, data) {
|
|
chrome.tabs.query({ active: true, currentWindow: true }, function (tabs) {
|
|
var tabId = null;
|
|
if (tabs.length > 0) {
|
|
tabId = tabs[0].id;
|
|
}
|
|
else {
|
|
return;
|
|
}
|
|
|
|
messageTab(tabId, command, data);
|
|
});
|
|
}
|
|
|
|
function messageTab(tabId, command, data) {
|
|
if (!tabId) {
|
|
return;
|
|
}
|
|
|
|
var obj = {
|
|
command: command
|
|
};
|
|
|
|
if (data) {
|
|
obj['data'] = data;
|
|
}
|
|
|
|
chrome.tabs.sendMessage(tabId, obj);
|
|
}
|
|
|
|
function collectPageDetailsForContentScript(tab) {
|
|
chrome.tabs.sendMessage(tab.id, { command: 'collectPageDetails', tabId: tab.id, contentScript: true }, function () { });
|
|
}
|
|
|
|
function addLogin(login, tab) {
|
|
var loginDomain = tldjs.getDomain(login.url);
|
|
if (!loginDomain) {
|
|
return;
|
|
}
|
|
|
|
loginService.getAllDecryptedForDomain(loginDomain).then(function (logins) {
|
|
var match = false;
|
|
for (var i = 0; i < logins.length; i++) {
|
|
if (logins[i].username === login.username) {
|
|
match = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!match) {
|
|
// remove any old logins for this tab
|
|
removeAddLogin(tab);
|
|
|
|
loginsToAdd.push({
|
|
username: login.username,
|
|
password: login.password,
|
|
name: loginDomain,
|
|
uri: login.url,
|
|
tabId: tab.id,
|
|
expires: new Date((new Date()).getTime() + 30 * 60000) // 30 minutes
|
|
});
|
|
checkLoginsToAdd(tab);
|
|
}
|
|
});
|
|
}
|
|
|
|
cleanupLoginsToAdd();
|
|
setInterval(cleanupLoginsToAdd, 2 * 60 * 1000); // check every 2 minutes
|
|
function cleanupLoginsToAdd() {
|
|
var now = new Date();
|
|
for (var i = loginsToAdd.length - 1; i >= 0 ; i--) {
|
|
if (loginsToAdd[i].expires < now) {
|
|
loginsToAdd.splice(i, 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
function removeAddLogin(tab) {
|
|
for (var i = loginsToAdd.length - 1; i >= 0 ; i--) {
|
|
if (loginsToAdd[i].tabId === tab.id) {
|
|
loginsToAdd.splice(i, 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
function saveAddLogin(tab) {
|
|
for (var i = loginsToAdd.length - 1; i >= 0 ; i--) {
|
|
if (loginsToAdd[i].tabId === tab.id) {
|
|
var loginToAdd = loginsToAdd[i];
|
|
loginsToAdd.splice(i, 1);
|
|
loginService.encrypt({
|
|
id: null,
|
|
folderId: null,
|
|
favorite: false,
|
|
name: loginToAdd.name,
|
|
uri: loginToAdd.uri,
|
|
username: loginToAdd.username,
|
|
password: loginToAdd.password,
|
|
notes: null
|
|
}).then(function (loginModel) {
|
|
var login = new Login(loginModel, true);
|
|
loginService.saveWithServer(login).then(function (login) {
|
|
ga('send', {
|
|
hitType: 'event',
|
|
eventAction: 'Added Login from Notification Bar'
|
|
});
|
|
});
|
|
});
|
|
messageTab(tab.id, 'closeNotificationBar');
|
|
}
|
|
}
|
|
}
|
|
|
|
function checkLoginsToAdd(tab) {
|
|
if (!loginsToAdd.length) {
|
|
return;
|
|
}
|
|
|
|
if (tab) {
|
|
check();
|
|
return;
|
|
}
|
|
|
|
chrome.tabs.query({ active: true, currentWindow: true }, function (tabs) {
|
|
if (tabs.length > 0) {
|
|
tab = tabs[0];
|
|
check();
|
|
}
|
|
});
|
|
|
|
function check() {
|
|
if (!tab) {
|
|
return;
|
|
}
|
|
|
|
var tabDomain = tldjs.getDomain(tab.url);
|
|
if (!tabDomain) {
|
|
return;
|
|
}
|
|
|
|
for (var i = 0; i < loginsToAdd.length; i++) {
|
|
// loginsToAdd[x].name is the domain here
|
|
if (loginsToAdd[i].tabId === tab.id && loginsToAdd[i].name === tabDomain) {
|
|
messageTab(tab.id, 'openNotificationBar', { type: 'add' });
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function startAutofillPage(login) {
|
|
loginToAutoFill = login;
|
|
chrome.tabs.query({ active: true, currentWindow: true }, function (tabs) {
|
|
var tabId = null;
|
|
if (tabs.length > 0) {
|
|
tabId = tabs[0].id;
|
|
}
|
|
else {
|
|
return;
|
|
}
|
|
|
|
if (!tabId) {
|
|
return;
|
|
}
|
|
|
|
chrome.tabs.sendMessage(tabId, { command: 'collectPageDetails', tabId: tabId }, function () { });
|
|
});
|
|
}
|
|
|
|
function autofillPage() {
|
|
chrome.tabs.query({ active: true, currentWindow: true }, function (tabs) {
|
|
var tabId = null;
|
|
if (tabs.length > 0) {
|
|
tabId = tabs[0].id;
|
|
}
|
|
else {
|
|
return;
|
|
}
|
|
|
|
if (!tabId) {
|
|
return;
|
|
}
|
|
|
|
if (loginToAutoFill && pageDetailsToAutoFill && pageDetailsToAutoFill.length) {
|
|
for (var i = 0; i < pageDetailsToAutoFill.length; i++) {
|
|
// make sure we're still on correct tab
|
|
if (pageDetailsToAutoFill[i].tabId !== tabId) {
|
|
continue;
|
|
}
|
|
|
|
var fillScript = autofillService.generateFillScript(pageDetailsToAutoFill[i].details, loginToAutoFill.username,
|
|
loginToAutoFill.password);
|
|
if (tabId && fillScript && fillScript.script && fillScript.script.length) {
|
|
chrome.tabs.sendMessage(tabId, {
|
|
command: 'fillForm',
|
|
fillScript: fillScript
|
|
}, {
|
|
frameId: pageDetailsToAutoFill[i].frameId
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
// reset
|
|
loginToAutoFill = null;
|
|
pageDetailsToAutoFill = [];
|
|
});
|
|
}
|
|
|
|
function sortLogins(logins) {
|
|
logins.sort(function (a, b) {
|
|
var nameA = (a.name + '_' + a.username).toUpperCase();
|
|
var nameB = (b.name + '_' + b.username).toUpperCase();
|
|
|
|
if (nameA < nameB) {
|
|
return -1;
|
|
}
|
|
if (nameA > nameB) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
});
|
|
}
|
|
|
|
function loadLoginContextMenuOptions(login) {
|
|
var title = login.name + (login.username && login.username !== '' ? ' (' + login.username + ')' : '');
|
|
loadContextMenuOptions(title, login.id, login);
|
|
}
|
|
|
|
function loadNoLoginsContextMenuOptions(noLoginsMessage) {
|
|
loadContextMenuOptions(noLoginsMessage, 'noop', null);
|
|
}
|
|
|
|
function loadContextMenuOptions(title, idSuffix, login) {
|
|
if (!login || (login.password && login.password !== '')) {
|
|
chrome.contextMenus.create({
|
|
type: 'normal',
|
|
id: 'autofill_' + idSuffix,
|
|
parentId: 'autofill',
|
|
contexts: ['all'],
|
|
title: title
|
|
});
|
|
}
|
|
|
|
if (!login || (login.username && login.username !== '')) {
|
|
chrome.contextMenus.create({
|
|
type: 'normal',
|
|
id: 'copy-username_' + idSuffix,
|
|
parentId: 'copy-username',
|
|
contexts: ['all'],
|
|
title: title
|
|
});
|
|
}
|
|
|
|
if (!login || (login.password && login.password !== '')) {
|
|
chrome.contextMenus.create({
|
|
type: 'normal',
|
|
id: 'copy-password_' + idSuffix,
|
|
parentId: 'copy-password',
|
|
contexts: ['all'],
|
|
title: title
|
|
});
|
|
}
|
|
}
|
|
|
|
function copyToClipboard(text) {
|
|
if (window.clipboardData && window.clipboardData.setData) {
|
|
// IE specific code path to prevent textarea being shown while dialog is visible.
|
|
return clipboardData.setData('Text', text);
|
|
}
|
|
else if (document.queryCommandSupported && document.queryCommandSupported('copy')) {
|
|
var textarea = document.createElement('textarea');
|
|
textarea.textContent = text;
|
|
// Prevent scrolling to bottom of page in MS Edge.
|
|
textarea.style.position = 'fixed';
|
|
document.body.appendChild(textarea);
|
|
textarea.select();
|
|
|
|
try {
|
|
// Security exception may be thrown by some browsers.
|
|
return document.execCommand('copy');
|
|
}
|
|
catch (ex) {
|
|
console.warn('Copy to clipboard failed.', ex);
|
|
return false;
|
|
}
|
|
finally {
|
|
document.body.removeChild(textarea);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Sync polling
|
|
|
|
fullSync(true);
|
|
setInterval(fullSync, 5 * 60 * 1000); // check every 5 minutes
|
|
var syncInternal = 6 * 60 * 60 * 1000; // 6 hours
|
|
|
|
function fullSync(override) {
|
|
override = override || false;
|
|
syncService.getLastSync(function (lastSync) {
|
|
var now = new Date();
|
|
if (override || !lastSync || (now - lastSync) >= syncInternal) {
|
|
syncService.fullSync(override || false, function () { });
|
|
}
|
|
});
|
|
}
|
|
|
|
// Locking
|
|
|
|
checkLock();
|
|
setInterval(checkLock, 10 * 1000); // check every 10 seconds
|
|
|
|
function checkLock() {
|
|
if (chrome.extension.getViews({ type: 'popup' }).length > 0) {
|
|
// popup is open, do not lock
|
|
return;
|
|
}
|
|
|
|
cryptoService.getKey(false, function (key) {
|
|
if (!key) {
|
|
// no key so no need to lock
|
|
return;
|
|
}
|
|
|
|
chrome.storage.local.get(constantsService.lockOptionKey, function (obj) {
|
|
if (obj && ((!obj[constantsService.lockOptionKey] && obj[constantsService.lockOptionKey] !== 0) ||
|
|
obj[constantsService.lockOptionKey] === -1)) {
|
|
// no lock option set
|
|
return;
|
|
}
|
|
|
|
chrome.storage.local.get(constantsService.lastActiveKey, function (obj2) {
|
|
if (obj2 && obj2[constantsService.lastActiveKey]) {
|
|
var lastActive = obj2[constantsService.lastActiveKey];
|
|
var diffSeconds = ((new Date()).getTime() - lastActive) / 1000;
|
|
var lockOptionSeconds = parseInt(obj[constantsService.lockOptionKey]) * 60;
|
|
|
|
if (diffSeconds >= lockOptionSeconds) {
|
|
// need to lock now
|
|
cryptoService.clearKey(function () {
|
|
setIcon();
|
|
folderService.clearCache();
|
|
loginService.clearCache();
|
|
refreshBadgeAndMenu();
|
|
});
|
|
}
|
|
}
|
|
});
|
|
});
|
|
});
|
|
};
|