This is imported from https://github.com/mozilla-services/screenshots/. It has been reviewed as patches landed, but also reviewed by Mossop and kmag. This also includes the patch from bug 1356394 MozReview-Commit-ID: FXIVw7WjxlN
217 lines
6.5 KiB
JavaScript
217 lines
6.5 KiB
JavaScript
/* globals browser, log */
|
|
/* globals main, makeUuid, deviceInfo, analytics, catcher, buildSettings, communication */
|
|
|
|
"use strict";
|
|
|
|
this.auth = (function () {
|
|
let exports = {};
|
|
|
|
let registrationInfo;
|
|
let initialized = false;
|
|
let authHeader = null;
|
|
let sentryPublicDSN = null;
|
|
let abTests = {};
|
|
|
|
catcher.watchPromise(browser.storage.local.get(["registrationInfo", "abTests"]).then((result) => {
|
|
if (result.abTests) {
|
|
abTests = result.abTests;
|
|
}
|
|
if (result.registrationInfo) {
|
|
registrationInfo = result.registrationInfo;
|
|
} else {
|
|
registrationInfo = generateRegistrationInfo();
|
|
log.info("Generating new device authentication ID", registrationInfo);
|
|
return browser.storage.local.set({registrationInfo});
|
|
}
|
|
}));
|
|
|
|
exports.getDeviceId = function () {
|
|
return registrationInfo && registrationInfo.deviceId;
|
|
};
|
|
|
|
function generateRegistrationInfo() {
|
|
let info = {
|
|
deviceId: `anon${makeUuid()}`,
|
|
secret: makeUuid(),
|
|
registered: false
|
|
};
|
|
return info;
|
|
}
|
|
|
|
function register() {
|
|
return new Promise((resolve, reject) => {
|
|
let registerUrl = main.getBackend() + "/api/register";
|
|
// TODO: replace xhr with Fetch #2261
|
|
let req = new XMLHttpRequest();
|
|
req.open("POST", registerUrl);
|
|
req.setRequestHeader("content-type", "application/json");
|
|
req.onload = catcher.watchFunction(() => {
|
|
if (req.status == 200) {
|
|
log.info("Registered login");
|
|
initialized = true;
|
|
saveAuthInfo(JSON.parse(req.responseText));
|
|
resolve(true);
|
|
analytics.sendEvent("registered");
|
|
} else {
|
|
analytics.sendEvent("register-failed", `bad-response-${req.status}`);
|
|
log.warn("Error in response:", req.responseText);
|
|
let exc = new Error("Bad response: " + req.status);
|
|
exc.popupMessage = "LOGIN_ERROR";
|
|
reject(exc);
|
|
}
|
|
});
|
|
req.onerror = catcher.watchFunction(() => {
|
|
analytics.sendEvent("register-failed", "connection-error");
|
|
let exc = new Error("Error contacting server");
|
|
exc.popupMessage = "LOGIN_CONNECTION_ERROR";
|
|
reject(exc);
|
|
});
|
|
req.send(JSON.stringify({
|
|
deviceId: registrationInfo.deviceId,
|
|
secret: registrationInfo.secret,
|
|
deviceInfo: JSON.stringify(deviceInfo())
|
|
}));
|
|
});
|
|
}
|
|
|
|
function login(options) {
|
|
let { ownershipCheck, noRegister } = options || {};
|
|
return new Promise((resolve, reject) => {
|
|
let loginUrl = main.getBackend() + "/api/login";
|
|
// TODO: replace xhr with Fetch #2261
|
|
let req = new XMLHttpRequest();
|
|
req.open("POST", loginUrl);
|
|
req.onload = catcher.watchFunction(() => {
|
|
if (req.status == 404) {
|
|
if (noRegister) {
|
|
resolve(false);
|
|
} else {
|
|
resolve(register());
|
|
}
|
|
} else if (req.status >= 300) {
|
|
log.warn("Error in response:", req.responseText);
|
|
let exc = new Error("Could not log in: " + req.status);
|
|
exc.popupMessage = "LOGIN_ERROR";
|
|
analytics.sendEvent("login-failed", `bad-response-${req.status}`);
|
|
reject(exc);
|
|
} else if (req.status === 0) {
|
|
let error = new Error("Could not log in, server unavailable");
|
|
error.popupMessage = "LOGIN_CONNECTION_ERROR";
|
|
analytics.sendEvent("login-failed", "connection-error");
|
|
reject(error);
|
|
} else {
|
|
initialized = true;
|
|
let jsonResponse = JSON.parse(req.responseText);
|
|
log.info("Screenshots logged in");
|
|
analytics.sendEvent("login");
|
|
saveAuthInfo(jsonResponse);
|
|
if (ownershipCheck) {
|
|
resolve({isOwner: jsonResponse.isOwner});
|
|
} else {
|
|
resolve(true);
|
|
}
|
|
}
|
|
});
|
|
req.onerror = catcher.watchFunction(() => {
|
|
analytics.sendEvent("login-failed", "connection-error");
|
|
let exc = new Error("Connection failed");
|
|
exc.url = loginUrl;
|
|
exc.popupMessage = "CONNECTION_ERROR";
|
|
reject(exc);
|
|
});
|
|
req.setRequestHeader("content-type", "application/json");
|
|
req.send(JSON.stringify({
|
|
deviceId: registrationInfo.deviceId,
|
|
secret: registrationInfo.secret,
|
|
deviceInfo: JSON.stringify(deviceInfo()),
|
|
ownershipCheck
|
|
}));
|
|
});
|
|
}
|
|
|
|
function saveAuthInfo(responseJson) {
|
|
if (responseJson.sentryPublicDSN) {
|
|
sentryPublicDSN = responseJson.sentryPublicDSN;
|
|
}
|
|
if (responseJson.authHeader) {
|
|
authHeader = responseJson.authHeader;
|
|
if (!registrationInfo.registered) {
|
|
registrationInfo.registered = true;
|
|
catcher.watchPromise(browser.storage.local.set({registrationInfo}));
|
|
}
|
|
}
|
|
if (responseJson.abTests) {
|
|
abTests = responseJson.abTests;
|
|
catcher.watchPromise(browser.storage.local.set({abTests}));
|
|
}
|
|
}
|
|
|
|
exports.getDeviceId = function () {
|
|
return registrationInfo.deviceId;
|
|
};
|
|
|
|
exports.authHeaders = function () {
|
|
let initPromise = Promise.resolve();
|
|
if (! initialized) {
|
|
initPromise = login();
|
|
}
|
|
return initPromise.then(() => {
|
|
if (authHeader) {
|
|
return {"x-screenshots-auth": authHeader};
|
|
} else {
|
|
log.warn("No auth header available");
|
|
return {};
|
|
}
|
|
});
|
|
};
|
|
|
|
exports.getSentryPublicDSN = function () {
|
|
return sentryPublicDSN || buildSettings.defaultSentryDsn;
|
|
};
|
|
|
|
exports.getAbTests = function () {
|
|
return abTests;
|
|
};
|
|
|
|
exports.isRegistered = function () {
|
|
return registrationInfo.registered;
|
|
};
|
|
|
|
exports.setDeviceInfoFromOldAddon = function (newDeviceInfo) {
|
|
if (! (newDeviceInfo.deviceId && newDeviceInfo.secret)) {
|
|
throw new Error("Bad deviceInfo");
|
|
}
|
|
if (registrationInfo.deviceId === newDeviceInfo.deviceId &&
|
|
registrationInfo.secret === newDeviceInfo.secret) {
|
|
// Probably we already imported the information
|
|
return Promise.resolve(false);
|
|
}
|
|
registrationInfo = {
|
|
deviceId: newDeviceInfo.deviceId,
|
|
secret: newDeviceInfo.secret,
|
|
registered: true
|
|
};
|
|
initialized = false;
|
|
return browser.storage.local.set({registrationInfo}).then(() => {
|
|
return true;
|
|
});
|
|
};
|
|
|
|
communication.register("getAuthInfo", (sender, ownershipCheck) => {
|
|
let info = registrationInfo;
|
|
let done = Promise.resolve();
|
|
if (info.registered) {
|
|
done = login({ownershipCheck}).then((result) => {
|
|
if (result && result.isOwner) {
|
|
info.isOwner = true;
|
|
}
|
|
});
|
|
}
|
|
return done.then(() => {
|
|
return info;
|
|
});
|
|
});
|
|
|
|
return exports;
|
|
})();
|