2016-05-28 05:05:27 +08:00
|
|
|
/* eslint global-require: 0 */
|
|
|
|
|
|
|
|
import crypto from 'crypto';
|
2017-09-27 02:33:08 +08:00
|
|
|
import { CommonProviderSettings } from 'imap-provider-settings';
|
2016-11-30 04:29:28 +08:00
|
|
|
import {
|
2017-09-11 13:45:48 +08:00
|
|
|
Account,
|
2017-09-27 02:46:00 +08:00
|
|
|
MailspringAPIRequest,
|
2017-08-07 05:18:04 +08:00
|
|
|
IdentityStore,
|
2016-11-30 04:29:28 +08:00
|
|
|
RegExpUtils,
|
2017-07-04 02:44:28 +08:00
|
|
|
MailsyncProcess,
|
2017-09-27 02:42:18 +08:00
|
|
|
} from 'mailspring-exports';
|
2016-11-24 03:48:58 +08:00
|
|
|
|
2017-09-27 02:46:00 +08:00
|
|
|
const { makeRequest, rootURLForServer } = MailspringAPIRequest;
|
2017-07-04 02:44:28 +08:00
|
|
|
|
2017-09-11 13:45:48 +08:00
|
|
|
function base64URL(inBuffer) {
|
2016-06-02 21:45:48 +08:00
|
|
|
let buffer;
|
2017-09-27 02:33:08 +08:00
|
|
|
if (typeof inBuffer === 'string') {
|
2016-06-02 21:45:48 +08:00
|
|
|
buffer = new Buffer(inBuffer);
|
|
|
|
} else if (inBuffer instanceof Buffer) {
|
|
|
|
buffer = inBuffer;
|
|
|
|
} else {
|
2017-09-27 02:33:08 +08:00
|
|
|
throw new Error(`${inBuffer} must be a string or Buffer`);
|
2016-06-02 21:45:48 +08:00
|
|
|
}
|
2017-09-27 02:33:08 +08:00
|
|
|
return buffer
|
|
|
|
.toString('base64')
|
|
|
|
.replace(/\+/g, '-') // Convert '+' to '-'
|
2016-05-28 05:05:27 +08:00
|
|
|
.replace(/\//g, '_'); // Convert '/' to '_'
|
|
|
|
}
|
|
|
|
|
2017-09-11 03:04:22 +08:00
|
|
|
function idForAccount(emailAddress, connectionSettings) {
|
|
|
|
// changing your connection security settings / ports shouldn't blow
|
|
|
|
// away everything and trash your metadata. Just look at critiical fields.
|
|
|
|
// (Me adding more connection settings fields shouldn't break account Ids either!)
|
|
|
|
const settingsThatCouldChangeMailContents = {
|
|
|
|
imap_username: connectionSettings.imap_username,
|
|
|
|
imap_host: connectionSettings.imap_host,
|
|
|
|
smtp_username: connectionSettings.smtp_username,
|
|
|
|
smtp_host: connectionSettings.smtp_host,
|
2017-09-27 02:33:08 +08:00
|
|
|
};
|
2017-09-11 03:04:22 +08:00
|
|
|
|
|
|
|
const idString = `${emailAddress}${JSON.stringify(settingsThatCouldChangeMailContents)}`;
|
2017-09-27 02:33:08 +08:00
|
|
|
return crypto
|
|
|
|
.createHash('sha256')
|
|
|
|
.update(idString, 'utf8')
|
|
|
|
.digest('hex')
|
|
|
|
.substr(0, 8);
|
2017-09-11 03:04:22 +08:00
|
|
|
}
|
|
|
|
|
2017-09-11 13:45:48 +08:00
|
|
|
export function expandAccountWithCommonSettings(account) {
|
2017-09-27 02:33:08 +08:00
|
|
|
const domain = account.emailAddress
|
|
|
|
.split('@')
|
|
|
|
.pop()
|
|
|
|
.toLowerCase();
|
2017-09-11 13:45:48 +08:00
|
|
|
let template = CommonProviderSettings[domain] || CommonProviderSettings[account.provider] || {};
|
|
|
|
if (template.alias) {
|
|
|
|
template = CommonProviderSettings[template.alias];
|
|
|
|
}
|
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
const usernameWithFormat = format => {
|
2017-09-11 13:45:48 +08:00
|
|
|
if (format === 'email') {
|
2017-09-27 02:33:08 +08:00
|
|
|
return account.emailAddress;
|
2017-09-11 13:45:48 +08:00
|
|
|
}
|
|
|
|
if (format === 'email-without-domain') {
|
|
|
|
return account.emailAddress.split('@').shift();
|
|
|
|
}
|
|
|
|
return undefined;
|
2017-09-27 02:33:08 +08:00
|
|
|
};
|
2017-09-11 13:45:48 +08:00
|
|
|
|
|
|
|
const populated = account.clone();
|
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
populated.settings = Object.assign(
|
|
|
|
{
|
|
|
|
imap_host: template.imap_host,
|
|
|
|
imap_port: template.imap_port || 993,
|
|
|
|
imap_username: usernameWithFormat(template.imap_user_format),
|
|
|
|
imap_password: populated.settings.imap_password,
|
|
|
|
imap_security: template.imap_security || 'SSL / TLS',
|
|
|
|
imap_allow_insecure_ssl: template.imap_allow_insecure_ssl || false,
|
|
|
|
smtp_host: template.smtp_host,
|
|
|
|
smtp_port: template.smtp_port || 587,
|
|
|
|
smtp_username: usernameWithFormat(template.smtp_user_format),
|
|
|
|
smtp_password: populated.settings.smtp_password || populated.settings.imap_password,
|
|
|
|
smtp_security: template.smtp_security || 'STARTTLS',
|
|
|
|
smtp_allow_insecure_ssl: template.smtp_allow_insecure_ssl || false,
|
|
|
|
},
|
|
|
|
populated.settings
|
|
|
|
);
|
2017-09-11 13:45:48 +08:00
|
|
|
|
|
|
|
return populated;
|
|
|
|
}
|
|
|
|
|
2017-08-01 11:20:01 +08:00
|
|
|
export function makeGmailOAuthRequest(sessionKey) {
|
|
|
|
return makeRequest({
|
2017-09-11 13:45:48 +08:00
|
|
|
server: 'identity',
|
2017-08-01 11:20:01 +08:00
|
|
|
path: `/auth/gmail/token?key=${sessionKey}`,
|
|
|
|
method: 'GET',
|
|
|
|
auth: false,
|
2016-05-28 05:05:27 +08:00
|
|
|
});
|
2017-01-19 09:44:22 +08:00
|
|
|
}
|
|
|
|
|
2017-09-11 13:45:48 +08:00
|
|
|
export async function buildGmailAccountFromToken(serverTokenResponse) {
|
2017-09-06 04:37:40 +08:00
|
|
|
// At this point, the Mailspring server has retrieved the Gmail token,
|
2017-08-30 03:17:45 +08:00
|
|
|
// created an account object in the database and tested it. All we
|
|
|
|
// need to do is save it locally, since we're confident Gmail will be
|
|
|
|
// accessible from the local sync worker.
|
2017-09-27 02:33:08 +08:00
|
|
|
const { name, emailAddress, refreshToken } = serverTokenResponse;
|
2017-09-26 06:44:20 +08:00
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
const account = expandAccountWithCommonSettings(
|
|
|
|
new Account({
|
|
|
|
name: name,
|
|
|
|
emailAddress: emailAddress,
|
|
|
|
provider: 'gmail',
|
|
|
|
settings: {
|
|
|
|
refresh_token: refreshToken,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
);
|
2017-09-26 06:44:20 +08:00
|
|
|
|
|
|
|
account.id = idForAccount(emailAddress, account.settings);
|
|
|
|
|
|
|
|
return account;
|
2016-05-28 05:05:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export function buildGmailSessionKey() {
|
2017-09-11 13:45:48 +08:00
|
|
|
return base64URL(crypto.randomBytes(40));
|
2016-05-28 05:05:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export function buildGmailAuthURL(sessionKey) {
|
2017-09-11 13:45:48 +08:00
|
|
|
return `${rootURLForServer('identity')}/auth/gmail?state=${sessionKey}`;
|
2016-05-28 05:05:27 +08:00
|
|
|
}
|
|
|
|
|
2017-09-11 13:45:48 +08:00
|
|
|
export async function finalizeAndValidateAccount(account) {
|
|
|
|
account.id = idForAccount(account.emailAddress, account.settings);
|
2016-05-28 05:05:27 +08:00
|
|
|
|
|
|
|
// handle special case for exchange/outlook/hotmail username field
|
2017-09-11 13:45:48 +08:00
|
|
|
account.settings.username = account.settings.username || account.settings.email;
|
2016-05-28 05:05:27 +08:00
|
|
|
|
2017-09-11 13:45:48 +08:00
|
|
|
if (account.settings.imap_port) {
|
|
|
|
account.settings.imap_port /= 1;
|
2016-05-28 05:05:27 +08:00
|
|
|
}
|
2017-09-11 13:45:48 +08:00
|
|
|
if (account.settings.smtp_port) {
|
|
|
|
account.settings.smtp_port /= 1;
|
2016-11-24 03:48:58 +08:00
|
|
|
}
|
|
|
|
|
2017-09-11 13:45:48 +08:00
|
|
|
// Test connections to IMAP and SMTP
|
2017-09-27 02:36:58 +08:00
|
|
|
const proc = new MailsyncProcess(AppEnv.getLoadSettings(), IdentityStore.identity(), account);
|
2017-09-11 13:45:48 +08:00
|
|
|
const response = await proc.test();
|
|
|
|
return new Account(response.account);
|
2016-05-28 05:05:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export function isValidHost(value) {
|
|
|
|
return RegExpUtils.domainRegex().test(value) || RegExpUtils.ipAddressRegex().test(value);
|
|
|
|
}
|