436 lines
12 KiB
JavaScript
436 lines
12 KiB
JavaScript
import i18n from "i18next";
|
|
import {
|
|
accountBillingPortalUrl,
|
|
accountBillingSubscriptionUrl,
|
|
accountPasswordUrl,
|
|
accountPhoneUrl,
|
|
accountPhoneVerifyUrl,
|
|
accountReservationSingleUrl,
|
|
accountReservationUrl,
|
|
accountSettingsUrl,
|
|
accountSubscriptionUrl,
|
|
accountTokenUrl,
|
|
accountUrl,
|
|
maybeWithBearerAuth,
|
|
tiersUrl,
|
|
withBasicAuth,
|
|
withBearerAuth,
|
|
} from "./utils";
|
|
import session from "./Session";
|
|
import subscriptionManager from "./SubscriptionManager";
|
|
import prefs from "./Prefs";
|
|
import routes from "../components/routes";
|
|
import { fetchOrThrow, UnauthorizedError } from "./errors";
|
|
|
|
const delayMillis = 45000; // 45 seconds
|
|
const intervalMillis = 900000; // 15 minutes
|
|
|
|
class AccountApi {
|
|
constructor() {
|
|
this.timer = null;
|
|
this.listener = null; // Fired when account is fetched from remote
|
|
this.tiers = null; // Cached
|
|
}
|
|
|
|
registerListener(listener) {
|
|
this.listener = listener;
|
|
}
|
|
|
|
resetListener() {
|
|
this.listener = null;
|
|
}
|
|
|
|
async login(user) {
|
|
const url = accountTokenUrl(config.base_url);
|
|
console.log(`[AccountApi] Checking auth for ${url}`);
|
|
const response = await fetchOrThrow(url, {
|
|
method: "POST",
|
|
headers: withBasicAuth({}, user.username, user.password),
|
|
});
|
|
const json = await response.json(); // May throw SyntaxError
|
|
if (!json.token) {
|
|
throw new Error(`Unexpected server response: Cannot find token`);
|
|
}
|
|
return json.token;
|
|
}
|
|
|
|
async logout() {
|
|
const url = accountTokenUrl(config.base_url);
|
|
console.log(`[AccountApi] Logging out from ${url} using token ${session.token()}`);
|
|
await fetchOrThrow(url, {
|
|
method: "DELETE",
|
|
headers: withBearerAuth({}, session.token()),
|
|
});
|
|
}
|
|
|
|
async create(username, password) {
|
|
const url = accountUrl(config.base_url);
|
|
const body = JSON.stringify({
|
|
username,
|
|
password,
|
|
});
|
|
console.log(`[AccountApi] Creating user account ${url}`);
|
|
await fetchOrThrow(url, {
|
|
method: "POST",
|
|
body,
|
|
});
|
|
}
|
|
|
|
async get() {
|
|
const url = accountUrl(config.base_url);
|
|
console.log(`[AccountApi] Fetching user account ${url}`);
|
|
const response = await fetchOrThrow(url, {
|
|
headers: maybeWithBearerAuth({}, session.token()), // GET /v1/account endpoint can be called by anonymous
|
|
});
|
|
const account = await response.json(); // May throw SyntaxError
|
|
console.log(`[AccountApi] Account`, account);
|
|
if (this.listener) {
|
|
this.listener(account);
|
|
}
|
|
return account;
|
|
}
|
|
|
|
async delete(password) {
|
|
const url = accountUrl(config.base_url);
|
|
console.log(`[AccountApi] Deleting user account ${url}`);
|
|
await fetchOrThrow(url, {
|
|
method: "DELETE",
|
|
headers: withBearerAuth({}, session.token()),
|
|
body: JSON.stringify({
|
|
password,
|
|
}),
|
|
});
|
|
}
|
|
|
|
async changePassword(currentPassword, newPassword) {
|
|
const url = accountPasswordUrl(config.base_url);
|
|
console.log(`[AccountApi] Changing account password ${url}`);
|
|
await fetchOrThrow(url, {
|
|
method: "POST",
|
|
headers: withBearerAuth({}, session.token()),
|
|
body: JSON.stringify({
|
|
password: currentPassword,
|
|
new_password: newPassword,
|
|
}),
|
|
});
|
|
}
|
|
|
|
async createToken(label, expires) {
|
|
const url = accountTokenUrl(config.base_url);
|
|
const body = {
|
|
label,
|
|
expires: expires > 0 ? Math.floor(Date.now() / 1000) + expires : 0,
|
|
};
|
|
console.log(`[AccountApi] Creating user access token ${url}`);
|
|
await fetchOrThrow(url, {
|
|
method: "POST",
|
|
headers: withBearerAuth({}, session.token()),
|
|
body: JSON.stringify(body),
|
|
});
|
|
}
|
|
|
|
async updateToken(token, label, expires) {
|
|
const url = accountTokenUrl(config.base_url);
|
|
const body = {
|
|
token,
|
|
label,
|
|
};
|
|
if (expires > 0) {
|
|
body.expires = Math.floor(Date.now() / 1000) + expires;
|
|
}
|
|
console.log(`[AccountApi] Creating user access token ${url}`);
|
|
await fetchOrThrow(url, {
|
|
method: "PATCH",
|
|
headers: withBearerAuth({}, session.token()),
|
|
body: JSON.stringify(body),
|
|
});
|
|
}
|
|
|
|
async extendToken() {
|
|
const url = accountTokenUrl(config.base_url);
|
|
console.log(`[AccountApi] Extending user access token ${url}`);
|
|
await fetchOrThrow(url, {
|
|
method: "PATCH",
|
|
headers: withBearerAuth({}, session.token()),
|
|
});
|
|
}
|
|
|
|
async deleteToken(token) {
|
|
const url = accountTokenUrl(config.base_url);
|
|
console.log(`[AccountApi] Deleting user access token ${url}`);
|
|
await fetchOrThrow(url, {
|
|
method: "DELETE",
|
|
headers: withBearerAuth({ "X-Token": token }, session.token()),
|
|
});
|
|
}
|
|
|
|
async updateSettings(payload) {
|
|
const url = accountSettingsUrl(config.base_url);
|
|
const body = JSON.stringify(payload);
|
|
console.log(`[AccountApi] Updating user account ${url}: ${body}`);
|
|
await fetchOrThrow(url, {
|
|
method: "PATCH",
|
|
headers: withBearerAuth({}, session.token()),
|
|
body,
|
|
});
|
|
}
|
|
|
|
async addSubscription(baseUrl, topic) {
|
|
const url = accountSubscriptionUrl(config.base_url);
|
|
const body = JSON.stringify({
|
|
base_url: baseUrl,
|
|
topic,
|
|
});
|
|
console.log(`[AccountApi] Adding user subscription ${url}: ${body}`);
|
|
const response = await fetchOrThrow(url, {
|
|
method: "POST",
|
|
headers: withBearerAuth({}, session.token()),
|
|
body,
|
|
});
|
|
const subscription = await response.json(); // May throw SyntaxError
|
|
console.log(`[AccountApi] Subscription`, subscription);
|
|
return subscription;
|
|
}
|
|
|
|
async updateSubscription(baseUrl, topic, payload) {
|
|
const url = accountSubscriptionUrl(config.base_url);
|
|
const body = JSON.stringify({
|
|
base_url: baseUrl,
|
|
topic,
|
|
...payload,
|
|
});
|
|
console.log(`[AccountApi] Updating user subscription ${url}: ${body}`);
|
|
const response = await fetchOrThrow(url, {
|
|
method: "PATCH",
|
|
headers: withBearerAuth({}, session.token()),
|
|
body,
|
|
});
|
|
const subscription = await response.json(); // May throw SyntaxError
|
|
console.log(`[AccountApi] Subscription`, subscription);
|
|
return subscription;
|
|
}
|
|
|
|
async deleteSubscription(baseUrl, topic) {
|
|
const url = accountSubscriptionUrl(config.base_url);
|
|
console.log(`[AccountApi] Removing user subscription ${url}`);
|
|
const headers = {
|
|
"X-BaseURL": baseUrl,
|
|
"X-Topic": topic,
|
|
};
|
|
await fetchOrThrow(url, {
|
|
method: "DELETE",
|
|
headers: withBearerAuth(headers, session.token()),
|
|
});
|
|
}
|
|
|
|
async upsertReservation(topic, everyone) {
|
|
const url = accountReservationUrl(config.base_url);
|
|
console.log(`[AccountApi] Upserting user access to topic ${topic}, everyone=${everyone}`);
|
|
await fetchOrThrow(url, {
|
|
method: "POST",
|
|
headers: withBearerAuth({}, session.token()),
|
|
body: JSON.stringify({
|
|
topic,
|
|
everyone,
|
|
}),
|
|
});
|
|
}
|
|
|
|
async deleteReservation(topic, deleteMessages) {
|
|
const url = accountReservationSingleUrl(config.base_url, topic);
|
|
console.log(`[AccountApi] Removing topic reservation ${url}`);
|
|
const headers = {
|
|
"X-Delete-Messages": deleteMessages ? "true" : "false",
|
|
};
|
|
await fetchOrThrow(url, {
|
|
method: "DELETE",
|
|
headers: withBearerAuth(headers, session.token()),
|
|
});
|
|
}
|
|
|
|
async billingTiers() {
|
|
if (this.tiers) {
|
|
return this.tiers;
|
|
}
|
|
const url = tiersUrl(config.base_url);
|
|
console.log(`[AccountApi] Fetching billing tiers`);
|
|
const response = await fetchOrThrow(url); // No auth needed!
|
|
this.tiers = await response.json(); // May throw SyntaxError
|
|
return this.tiers;
|
|
}
|
|
|
|
async createBillingSubscription(tier, interval) {
|
|
console.log(`[AccountApi] Creating billing subscription with ${tier} and interval ${interval}`);
|
|
return this.upsertBillingSubscription("POST", tier, interval);
|
|
}
|
|
|
|
async updateBillingSubscription(tier, interval) {
|
|
console.log(`[AccountApi] Updating billing subscription with ${tier} and interval ${interval}`);
|
|
return this.upsertBillingSubscription("PUT", tier, interval);
|
|
}
|
|
|
|
async upsertBillingSubscription(method, tier, interval) {
|
|
const url = accountBillingSubscriptionUrl(config.base_url);
|
|
const response = await fetchOrThrow(url, {
|
|
method,
|
|
headers: withBearerAuth({}, session.token()),
|
|
body: JSON.stringify({
|
|
tier,
|
|
interval,
|
|
}),
|
|
});
|
|
return response.json(); // May throw SyntaxError
|
|
}
|
|
|
|
async deleteBillingSubscription() {
|
|
const url = accountBillingSubscriptionUrl(config.base_url);
|
|
console.log(`[AccountApi] Cancelling billing subscription`);
|
|
await fetchOrThrow(url, {
|
|
method: "DELETE",
|
|
headers: withBearerAuth({}, session.token()),
|
|
});
|
|
}
|
|
|
|
async createBillingPortalSession() {
|
|
const url = accountBillingPortalUrl(config.base_url);
|
|
console.log(`[AccountApi] Creating billing portal session`);
|
|
const response = await fetchOrThrow(url, {
|
|
method: "POST",
|
|
headers: withBearerAuth({}, session.token()),
|
|
});
|
|
return response.json(); // May throw SyntaxError
|
|
}
|
|
|
|
async verifyPhoneNumber(phoneNumber, channel) {
|
|
const url = accountPhoneVerifyUrl(config.base_url);
|
|
console.log(`[AccountApi] Sending phone verification ${url}`);
|
|
await fetchOrThrow(url, {
|
|
method: "PUT",
|
|
headers: withBearerAuth({}, session.token()),
|
|
body: JSON.stringify({
|
|
number: phoneNumber,
|
|
channel,
|
|
}),
|
|
});
|
|
}
|
|
|
|
async addPhoneNumber(phoneNumber, code) {
|
|
const url = accountPhoneUrl(config.base_url);
|
|
console.log(`[AccountApi] Adding phone number with verification code ${url}`);
|
|
await fetchOrThrow(url, {
|
|
method: "PUT",
|
|
headers: withBearerAuth({}, session.token()),
|
|
body: JSON.stringify({
|
|
number: phoneNumber,
|
|
code,
|
|
}),
|
|
});
|
|
}
|
|
|
|
async deletePhoneNumber(phoneNumber) {
|
|
const url = accountPhoneUrl(config.base_url);
|
|
console.log(`[AccountApi] Deleting phone number ${url}`);
|
|
await fetchOrThrow(url, {
|
|
method: "DELETE",
|
|
headers: withBearerAuth({}, session.token()),
|
|
body: JSON.stringify({
|
|
number: phoneNumber,
|
|
}),
|
|
});
|
|
}
|
|
|
|
async sync() {
|
|
try {
|
|
if (!session.token()) {
|
|
return null;
|
|
}
|
|
console.log(`[AccountApi] Syncing account`);
|
|
const account = await this.get();
|
|
if (account.language) {
|
|
await i18n.changeLanguage(account.language);
|
|
}
|
|
if (account.notification) {
|
|
if (account.notification.sound) {
|
|
await prefs.setSound(account.notification.sound);
|
|
}
|
|
if (account.notification.delete_after) {
|
|
await prefs.setDeleteAfter(account.notification.delete_after);
|
|
}
|
|
if (account.notification.min_priority) {
|
|
await prefs.setMinPriority(account.notification.min_priority);
|
|
}
|
|
}
|
|
if (account.subscriptions) {
|
|
await subscriptionManager.syncFromRemote(account.subscriptions, account.reservations);
|
|
}
|
|
return account;
|
|
} catch (e) {
|
|
console.log(`[AccountApi] Error fetching account`, e);
|
|
if (e instanceof UnauthorizedError) {
|
|
await session.resetAndRedirect(routes.login);
|
|
}
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
startWorker() {
|
|
if (this.timer !== null) {
|
|
return;
|
|
}
|
|
console.log(`[AccountApi] Starting worker`);
|
|
this.timer = setInterval(() => this.runWorker(), intervalMillis);
|
|
setTimeout(() => this.runWorker(), delayMillis);
|
|
}
|
|
|
|
stopWorker() {
|
|
clearTimeout(this.timer);
|
|
}
|
|
|
|
async runWorker() {
|
|
if (!session.token()) {
|
|
return;
|
|
}
|
|
console.log(`[AccountApi] Extending user access token`);
|
|
try {
|
|
await this.extendToken();
|
|
} catch (e) {
|
|
console.log(`[AccountApi] Error extending user access token`, e);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Maps to user.Role in user/types.go
|
|
export const Role = {
|
|
ADMIN: "admin",
|
|
USER: "user",
|
|
};
|
|
|
|
// Maps to server.visitorLimitBasis in server/visitor.go
|
|
export const LimitBasis = {
|
|
IP: "ip",
|
|
TIER: "tier",
|
|
};
|
|
|
|
// Maps to stripe.SubscriptionStatus
|
|
export const SubscriptionStatus = {
|
|
ACTIVE: "active",
|
|
PAST_DUE: "past_due",
|
|
};
|
|
|
|
// Maps to stripe.PriceRecurringInterval
|
|
export const SubscriptionInterval = {
|
|
MONTH: "month",
|
|
YEAR: "year",
|
|
};
|
|
|
|
// Maps to user.Permission in user/types.go
|
|
export const Permission = {
|
|
READ_WRITE: "read-write",
|
|
READ_ONLY: "read-only",
|
|
WRITE_ONLY: "write-only",
|
|
DENY_ALL: "deny-all",
|
|
};
|
|
|
|
const accountApi = new AccountApi();
|
|
export default accountApi;
|