2013-06-06 07:34:22 -04:00
|
|
|
/**
|
|
|
|
* The main application controller
|
|
|
|
*/
|
2013-08-20 09:19:13 -04:00
|
|
|
define(function(require) {
|
2013-08-20 13:48:49 -04:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
var $ = require('jquery'),
|
2013-09-19 12:35:12 -04:00
|
|
|
ImapClient = require('imap-client'),
|
|
|
|
SmtpClient = require('smtp-client'),
|
2013-08-20 13:48:49 -04:00
|
|
|
EmailDAO = require('js/dao/email-dao'),
|
2013-10-29 07:19:27 -04:00
|
|
|
RestDAO = require('js/dao/rest-dao'),
|
|
|
|
PublicKeyDAO = require('js/dao/publickey-dao'),
|
|
|
|
LawnchairDAO = require('js/dao/lawnchair-dao'),
|
2013-08-20 13:48:49 -04:00
|
|
|
KeychainDAO = require('js/dao/keychain-dao'),
|
2013-09-26 07:26:57 -04:00
|
|
|
DeviceStorageDAO = require('js/dao/devicestorage-dao'),
|
2013-11-19 10:14:48 -05:00
|
|
|
InvitationDAO = require('js/dao/invitation-dao'),
|
|
|
|
OutboxBO = require('js/bo/outbox'),
|
2013-10-11 21:19:01 -04:00
|
|
|
PGP = require('js/crypto/pgp'),
|
2013-09-15 09:13:19 -04:00
|
|
|
config = require('js/app-config').config;
|
2013-08-20 13:48:49 -04:00
|
|
|
require('cordova');
|
|
|
|
|
|
|
|
var self = {};
|
|
|
|
|
|
|
|
/**
|
2013-10-09 10:40:36 -04:00
|
|
|
* Start the application
|
2013-08-20 13:48:49 -04:00
|
|
|
*/
|
|
|
|
self.start = function(callback) {
|
|
|
|
// are we running in native app or in browser?
|
|
|
|
if (document.URL.indexOf("http") === 0 || document.URL.indexOf("app") === 0 || document.URL.indexOf("chrome") === 0) {
|
|
|
|
console.log('Assuming Browser environment...');
|
|
|
|
onDeviceReady();
|
|
|
|
} else {
|
|
|
|
console.log('Assuming Cordova environment...');
|
|
|
|
document.addEventListener("deviceready", onDeviceReady, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
function onDeviceReady() {
|
|
|
|
console.log('Starting app.');
|
2013-10-09 10:40:36 -04:00
|
|
|
// init app config storage
|
2013-10-29 07:19:27 -04:00
|
|
|
self._appConfigStore = new DeviceStorageDAO(new LawnchairDAO());
|
2013-10-09 10:40:36 -04:00
|
|
|
self._appConfigStore.init('app-config', callback);
|
2013-08-20 13:48:49 -04:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-11-08 18:30:45 -05:00
|
|
|
self.checkForUpdate = function() {
|
2013-11-08 15:35:30 -05:00
|
|
|
if (!chrome || !chrome.runtime || !chrome.runtime.onUpdateAvailable) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-11-08 18:30:45 -05:00
|
|
|
// check for update and restart
|
2013-11-08 13:08:34 -05:00
|
|
|
chrome.runtime.onUpdateAvailable.addListener(function(details) {
|
2013-11-08 18:30:45 -05:00
|
|
|
console.log("Updating to version " + details.version);
|
|
|
|
chrome.runtime.reload();
|
2013-11-08 13:08:34 -05:00
|
|
|
});
|
|
|
|
chrome.runtime.requestUpdateCheck(function(status) {
|
|
|
|
if (status === "update_found") {
|
|
|
|
console.log("Update pending...");
|
|
|
|
} else if (status === "no_update") {
|
|
|
|
console.log("No update found.");
|
|
|
|
} else if (status === "throttled") {
|
|
|
|
console.log("Checking updates too frequently.");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2013-09-26 09:48:32 -04:00
|
|
|
/**
|
|
|
|
* Request an OAuth token from chrome for gmail users
|
|
|
|
*/
|
2013-10-21 07:10:42 -04:00
|
|
|
self.fetchOAuthToken = function(callback) {
|
2013-08-20 13:48:49 -04:00
|
|
|
// get OAuth Token from chrome
|
|
|
|
chrome.identity.getAuthToken({
|
|
|
|
'interactive': true
|
|
|
|
},
|
|
|
|
function(token) {
|
2013-11-12 14:00:53 -05:00
|
|
|
if ((chrome && chrome.runtime && chrome.runtime.lastError) || !token) {
|
2013-09-26 09:48:32 -04:00
|
|
|
callback({
|
2013-11-12 14:00:53 -05:00
|
|
|
errMsg: 'Error fetching an OAuth token for the user!',
|
|
|
|
err: chrome.runtime.lastError
|
2013-09-26 09:48:32 -04:00
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get email address for the token
|
|
|
|
self.queryEmailAddress(token, function(err, emailAddress) {
|
|
|
|
if (err || !emailAddress) {
|
2013-08-20 13:48:49 -04:00
|
|
|
callback({
|
2013-09-26 09:48:32 -04:00
|
|
|
errMsg: 'Error looking up email address on login!',
|
2013-08-20 13:48:49 -04:00
|
|
|
err: err
|
|
|
|
});
|
2013-09-26 09:48:32 -04:00
|
|
|
return;
|
2013-08-20 13:48:49 -04:00
|
|
|
}
|
|
|
|
|
2013-10-12 13:39:09 -04:00
|
|
|
// init the email dao
|
2013-10-21 07:10:42 -04:00
|
|
|
callback(null, {
|
|
|
|
emailAddress: emailAddress,
|
|
|
|
token: token
|
|
|
|
});
|
2013-09-26 09:48:32 -04:00
|
|
|
});
|
2013-08-20 13:48:49 -04:00
|
|
|
}
|
|
|
|
);
|
2013-09-04 15:01:32 -04:00
|
|
|
};
|
2013-08-20 13:48:49 -04:00
|
|
|
|
2013-09-26 09:48:32 -04:00
|
|
|
/**
|
|
|
|
* Lookup the user's email address. Check local cache if available, otherwise query google's token info api to learn the user's email address
|
|
|
|
*/
|
|
|
|
self.queryEmailAddress = function(token, callback) {
|
2013-10-09 10:40:36 -04:00
|
|
|
var itemKey = 'emailaddress';
|
2013-09-26 11:17:47 -04:00
|
|
|
|
2013-10-09 10:40:36 -04:00
|
|
|
self._appConfigStore.listItems(itemKey, 0, null, function(err, cachedItems) {
|
|
|
|
if (err) {
|
|
|
|
callback(err);
|
|
|
|
return;
|
|
|
|
}
|
2013-09-26 11:17:47 -04:00
|
|
|
|
2013-10-09 10:40:36 -04:00
|
|
|
// do roundtrip to google api if no email address is cached yet
|
|
|
|
if (!cachedItems || cachedItems.length < 1) {
|
|
|
|
queryGoogleApi();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
callback(null, cachedItems[0]);
|
2013-09-26 09:48:32 -04:00
|
|
|
});
|
2013-09-26 11:17:47 -04:00
|
|
|
|
|
|
|
function queryGoogleApi() {
|
|
|
|
// fetch gmail user's email address from the Google Authorization Server endpoint
|
|
|
|
$.ajax({
|
|
|
|
url: 'https://www.googleapis.com/oauth2/v1/tokeninfo?access_token=' + token,
|
|
|
|
type: 'GET',
|
|
|
|
dataType: 'json',
|
|
|
|
success: function(info) {
|
|
|
|
if (!info || !info.email) {
|
|
|
|
callback({
|
|
|
|
errMsg: 'Error looking up email address on google api!'
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// cache the email address on the device
|
2013-10-09 10:40:36 -04:00
|
|
|
self._appConfigStore.storeList([info.email], itemKey, function(err) {
|
2013-09-26 11:17:47 -04:00
|
|
|
callback(err, info.email);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
error: function(xhr, textStatus, err) {
|
|
|
|
callback({
|
|
|
|
errMsg: xhr.status + ': ' + xhr.statusText,
|
|
|
|
err: err
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2013-09-26 09:48:32 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Instanciate the mail email data access object and its dependencies. Login to imap on init.
|
|
|
|
*/
|
2013-10-21 07:10:42 -04:00
|
|
|
self.init = function(userId, token, callback) {
|
2013-11-14 11:41:31 -05:00
|
|
|
var auth, imapOptions, smtpOptions, certificate,
|
2013-11-19 10:14:48 -05:00
|
|
|
lawnchairDao, restDao, pubkeyDao, invitationDao,
|
2013-11-14 11:41:31 -05:00
|
|
|
keychain, imapClient, smtpClient, pgp, userStorage, xhr;
|
|
|
|
|
|
|
|
// fetch pinned local ssl certificate
|
|
|
|
xhr = new XMLHttpRequest();
|
|
|
|
xhr.open('GET', '/ca/Google_Internet_Authority_G2.pem');
|
|
|
|
xhr.onload = function() {
|
|
|
|
if (xhr.readyState === 4 && xhr.status === 200 && xhr.responseText) {
|
|
|
|
certificate = xhr.responseText;
|
|
|
|
setupDaos();
|
|
|
|
} else {
|
|
|
|
callback({
|
|
|
|
errMsg: 'Could not fetch pinned certificate!'
|
|
|
|
});
|
2013-08-20 13:48:49 -04:00
|
|
|
}
|
|
|
|
};
|
2013-11-14 11:41:31 -05:00
|
|
|
xhr.onerror = function() {
|
|
|
|
callback({
|
|
|
|
errMsg: 'Could not fetch pinned certificate!'
|
|
|
|
});
|
2013-08-20 13:48:49 -04:00
|
|
|
};
|
2013-11-14 11:41:31 -05:00
|
|
|
xhr.send();
|
|
|
|
|
|
|
|
function setupDaos() {
|
|
|
|
// create mail credentials objects for imap/smtp
|
|
|
|
auth = {
|
|
|
|
XOAuth2: {
|
|
|
|
user: userId,
|
|
|
|
clientId: config.gmail.clientId,
|
|
|
|
accessToken: token
|
|
|
|
}
|
|
|
|
};
|
|
|
|
imapOptions = {
|
|
|
|
secure: config.gmail.imap.secure,
|
|
|
|
port: config.gmail.imap.port,
|
|
|
|
host: config.gmail.imap.host,
|
|
|
|
auth: auth,
|
|
|
|
ca: [certificate]
|
|
|
|
};
|
|
|
|
smtpOptions = {
|
|
|
|
secure: config.gmail.smtp.secure,
|
|
|
|
port: config.gmail.smtp.port,
|
|
|
|
host: config.gmail.smtp.host,
|
2013-11-19 05:28:29 -05:00
|
|
|
auth: auth,
|
|
|
|
ca: [certificate]
|
2013-11-14 11:41:31 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
// init objects and inject dependencies
|
|
|
|
restDao = new RestDAO();
|
|
|
|
pubkeyDao = new PublicKeyDAO(restDao);
|
|
|
|
lawnchairDao = new LawnchairDAO();
|
|
|
|
keychain = new KeychainDAO(lawnchairDao, pubkeyDao);
|
|
|
|
imapClient = new ImapClient(imapOptions);
|
|
|
|
smtpClient = new SmtpClient(smtpOptions);
|
|
|
|
pgp = new PGP();
|
|
|
|
userStorage = new DeviceStorageDAO(lawnchairDao);
|
|
|
|
self._emailDao = new EmailDAO(keychain, imapClient, smtpClient, pgp, userStorage);
|
|
|
|
|
2013-11-19 10:14:48 -05:00
|
|
|
invitationDao = new InvitationDAO(restDao);
|
|
|
|
self._outboxBo = new OutboxBO(self._emailDao, invitationDao);
|
|
|
|
|
2013-11-14 11:41:31 -05:00
|
|
|
// init email dao
|
|
|
|
var account = {
|
|
|
|
emailAddress: userId,
|
|
|
|
asymKeySize: config.asymKeySize
|
|
|
|
};
|
|
|
|
|
|
|
|
self._emailDao.init(account, callback);
|
|
|
|
}
|
2013-08-20 13:48:49 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
return self;
|
2013-06-10 11:57:33 -04:00
|
|
|
});
|