Files
notesnook/packages/core/models/user.js
2020-11-17 12:17:56 +05:00

162 lines
3.6 KiB
JavaScript

import { EV } from "../common";
import Constants from "../utils/constants";
export default class User {
/**
*
* @param {import("../api").default} db
*/
constructor(db) {
this._db = db;
this._context = db.context;
}
async sync() {
var user = await this.get();
if (!user) return;
try {
var serverUser = await authRequest.call(
this,
"users",
undefined,
true,
true
);
} catch (e) {
if (e.message.includes("not authorized")) await this.logout();
else throw e;
}
await this.set({
...user,
...serverUser,
});
// propogate event
EV.publish("user:synced", user);
}
get() {
return this._context.read("user");
}
async key() {
const user = await this.get();
if (!user) return;
const key = await this._context.getCryptoKey(`_uk_@${user.username}`);
return { key, salt: user.salt };
}
async set(user) {
if (!user) return;
user = { ...(await this.get()), ...user };
await this._context.write(`user`, user);
}
async login(username, password) {
let response = await authRequest("oauth/token", {
username,
password,
grant_type: "password",
});
if (!response) return;
await this._postLogin(password, response);
}
async token() {
let user = await this.get();
if (!user) return;
if (!user.accessToken) {
return await this._context.remove("user");
}
if (user.expiry > Date.now()) {
return user.accessToken;
}
let response = await authRequest("oauth/token", {
refresh_token: user.refreshToken,
grant_type: "refresh_token",
});
if (!response) return;
user = {
...user,
accessToken: response.access_token,
refreshToken: response.refresh_token,
expiry: Date.now() + response.expiry * 100,
};
await this._context.write("user", user);
// propogate event
EV.publish("user:tokenRefreshed", user);
}
logout() {
// propogate event
EV.publish("user:loggedOut");
return this._context.clear();
}
async signup(username, email, password) {
let response = await authRequest("auth/register", {
username,
password,
email,
});
if (!response) return;
await this._postLogin(password, response);
}
async _postLogin(password, response) {
await this._context.deriveCryptoKey(`_uk_@${response.payload.username}`, {
password,
salt: response.payload.salt,
});
let user = userFromResponse(response);
await this._context.write("user", user);
// propogate event
EV.publish("user:loggedIn", user);
}
}
function userFromResponse(response) {
let user = {
...response.payload,
accessToken: response.access_token,
refreshToken: response.refresh_token,
expiry: Date.now() + response.expiry * 100,
};
return user;
}
async function authRequest(endpoint, data, auth = false, get = false) {
var headers = {};
if (auth) {
const token = await this.token();
headers = {
Authorization: `Bearer ${token}`,
};
}
let response = await fetch(`${Constants.HOST}/${endpoint}`, {
method: get ? "GET" : "POST",
headers: { ...Constants.HEADERS, ...headers },
body: get ? undefined : JSON.stringify(data),
});
if (response.ok) {
let result = await response.json();
if (result.error) {
throw new Error(result.error);
}
return result;
}
let json = await response.json();
let error =
json.error ||
`Request failed with status code: ${response.status} ${response.statusText}.`;
throw new Error(error);
}