2021-06-18 17:47:25 -04:00
|
|
|
import { openDB } from 'idb';
|
2021-06-18 14:25:33 -04:00
|
|
|
import { Client } from "revolt.js";
|
2021-06-19 13:46:05 -04:00
|
|
|
import { takeError } from "./util";
|
2021-06-18 15:07:26 -04:00
|
|
|
import { createContext } from "preact";
|
|
|
|
import { Children } from "../../types/Preact";
|
2021-07-02 06:13:14 -04:00
|
|
|
import { useHistory } from 'react-router-dom';
|
2021-06-18 15:07:26 -04:00
|
|
|
import { Route } from "revolt.js/dist/api/routes";
|
|
|
|
import { connectState } from "../../redux/connector";
|
2021-06-18 17:47:25 -04:00
|
|
|
import Preloader from "../../components/ui/Preloader";
|
2021-06-18 15:07:26 -04:00
|
|
|
import { WithDispatcher } from "../../redux/reducers";
|
|
|
|
import { AuthState } from "../../redux/reducers/auth";
|
2021-06-20 12:31:53 -04:00
|
|
|
import { useEffect, useMemo, useState } from "preact/hooks";
|
2021-06-21 08:28:26 -04:00
|
|
|
import { useIntermediate } from '../intermediate/Intermediate';
|
2021-06-18 17:47:25 -04:00
|
|
|
import { registerEvents, setReconnectDisallowed } from "./events";
|
2021-06-20 12:31:53 -04:00
|
|
|
import { SingletonMessageRenderer } from '../../lib/renderer/Singleton';
|
2021-06-18 12:57:08 -04:00
|
|
|
|
|
|
|
export enum ClientStatus {
|
2021-06-18 17:47:25 -04:00
|
|
|
INIT,
|
2021-06-18 12:57:08 -04:00
|
|
|
LOADING,
|
|
|
|
READY,
|
|
|
|
OFFLINE,
|
|
|
|
DISCONNECTED,
|
|
|
|
CONNECTING,
|
|
|
|
RECONNECTING,
|
2021-06-18 14:25:33 -04:00
|
|
|
ONLINE,
|
2021-06-18 12:57:08 -04:00
|
|
|
}
|
|
|
|
|
2021-06-18 15:07:26 -04:00
|
|
|
export interface ClientOperations {
|
|
|
|
login: (data: Route<"POST", "/auth/login">["data"]) => Promise<void>;
|
|
|
|
logout: (shouldRequest?: boolean) => Promise<void>;
|
|
|
|
loggedIn: () => boolean;
|
|
|
|
ready: () => boolean;
|
2021-07-02 06:13:14 -04:00
|
|
|
|
|
|
|
openDM: (user_id: string) => Promise<string>;
|
2021-06-18 15:07:26 -04:00
|
|
|
}
|
|
|
|
|
2021-07-04 07:09:39 -04:00
|
|
|
// TODO: remove temporary non-null assertions and properly typecheck these as they aren't always immedietely initialized
|
|
|
|
export const AppContext = createContext<Client>(null!);
|
|
|
|
export const StatusContext = createContext<ClientStatus>(null!);
|
|
|
|
export const OperationsContext = createContext<ClientOperations>(null!);
|
2021-06-18 15:07:26 -04:00
|
|
|
|
|
|
|
type Props = WithDispatcher & {
|
|
|
|
auth: AuthState;
|
|
|
|
children: Children;
|
|
|
|
};
|
|
|
|
|
2021-06-23 09:52:33 -04:00
|
|
|
function Context({ auth, children, dispatcher }: Props) {
|
2021-07-02 06:13:14 -04:00
|
|
|
const history = useHistory();
|
2021-06-21 08:28:26 -04:00
|
|
|
const { openScreen } = useIntermediate();
|
2021-06-18 17:47:25 -04:00
|
|
|
const [status, setStatus] = useState(ClientStatus.INIT);
|
|
|
|
const [client, setClient] = useState<Client>(undefined as unknown as Client);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
(async () => {
|
|
|
|
let db;
|
|
|
|
try {
|
2021-06-22 09:37:38 -04:00
|
|
|
// Match sw.ts#L23
|
2021-06-18 17:47:25 -04:00
|
|
|
db = await openDB('state', 3, {
|
|
|
|
upgrade(db) {
|
|
|
|
for (let store of [ "channels", "servers", "users", "members" ]) {
|
|
|
|
db.createObjectStore(store, {
|
|
|
|
keyPath: '_id'
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
} catch (err) {
|
|
|
|
console.error('Failed to open IndexedDB store, continuing without.');
|
|
|
|
}
|
|
|
|
|
2021-06-20 12:31:53 -04:00
|
|
|
const client = new Client({
|
2021-06-18 17:47:25 -04:00
|
|
|
autoReconnect: false,
|
|
|
|
apiURL: import.meta.env.VITE_API_URL,
|
|
|
|
debug: import.meta.env.DEV,
|
|
|
|
db
|
2021-06-20 12:31:53 -04:00
|
|
|
});
|
2021-06-18 17:47:25 -04:00
|
|
|
|
2021-06-20 12:31:53 -04:00
|
|
|
setClient(client);
|
|
|
|
SingletonMessageRenderer.subscribe(client);
|
2021-06-18 17:47:25 -04:00
|
|
|
setStatus(ClientStatus.LOADING);
|
|
|
|
})();
|
|
|
|
}, [ ]);
|
|
|
|
|
|
|
|
if (status === ClientStatus.INIT) return null;
|
2021-06-18 15:07:26 -04:00
|
|
|
|
2021-06-19 13:46:05 -04:00
|
|
|
const operations: ClientOperations = useMemo(() => {
|
|
|
|
return {
|
2021-06-18 17:47:25 -04:00
|
|
|
login: async data => {
|
|
|
|
setReconnectDisallowed(true);
|
|
|
|
|
|
|
|
try {
|
|
|
|
const onboarding = await client.login(data);
|
|
|
|
setReconnectDisallowed(false);
|
|
|
|
const login = () =>
|
|
|
|
dispatcher({
|
|
|
|
type: "LOGIN",
|
2021-07-04 07:09:39 -04:00
|
|
|
session: client.session! // TODO: verify that this null assertion is correct
|
2021-06-18 17:47:25 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
if (onboarding) {
|
2021-06-21 08:28:26 -04:00
|
|
|
openScreen({
|
2021-06-18 17:47:25 -04:00
|
|
|
id: "onboarding",
|
2021-07-04 07:09:39 -04:00
|
|
|
callback: (username: string) =>
|
|
|
|
onboarding(username, true).then(login)
|
2021-06-21 08:28:26 -04:00
|
|
|
});
|
2021-06-18 17:47:25 -04:00
|
|
|
} else {
|
|
|
|
login();
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
setReconnectDisallowed(false);
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
logout: async shouldRequest => {
|
|
|
|
dispatcher({ type: "LOGOUT" });
|
|
|
|
|
2021-06-29 17:21:26 -04:00
|
|
|
client.reset();
|
2021-06-18 17:47:25 -04:00
|
|
|
dispatcher({ type: "RESET" });
|
|
|
|
|
2021-06-21 08:28:26 -04:00
|
|
|
openScreen({ id: "none" });
|
2021-06-18 17:47:25 -04:00
|
|
|
setStatus(ClientStatus.READY);
|
|
|
|
|
|
|
|
client.websocket.disconnect();
|
|
|
|
|
|
|
|
if (shouldRequest) {
|
|
|
|
try {
|
|
|
|
await client.logout();
|
|
|
|
} catch (err) {
|
|
|
|
console.error(err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
loggedIn: () => typeof auth.active !== "undefined",
|
|
|
|
ready: () => (
|
2021-06-19 13:46:05 -04:00
|
|
|
operations.loggedIn() &&
|
2021-06-18 17:47:25 -04:00
|
|
|
typeof client.user !== "undefined"
|
2021-07-02 06:13:14 -04:00
|
|
|
),
|
|
|
|
openDM: async (user_id: string) => {
|
|
|
|
let channel = await client.users.openDM(user_id);
|
|
|
|
history.push(`/channel/${channel!._id}`);
|
|
|
|
return channel!._id;
|
|
|
|
}
|
2021-06-18 15:07:26 -04:00
|
|
|
}
|
2021-06-19 13:46:05 -04:00
|
|
|
}, [ client, auth.active ]);
|
2021-06-18 15:07:26 -04:00
|
|
|
|
2021-06-20 12:31:53 -04:00
|
|
|
useEffect(() => registerEvents({ operations, dispatcher }, setStatus, client), [ client ]);
|
2021-06-18 17:47:25 -04:00
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
(async () => {
|
2021-06-20 05:17:34 -04:00
|
|
|
if (client.db) {
|
|
|
|
await client.restore();
|
|
|
|
}
|
2021-06-18 17:47:25 -04:00
|
|
|
|
|
|
|
if (auth.active) {
|
|
|
|
dispatcher({ type: "QUEUE_FAIL_ALL" });
|
|
|
|
|
|
|
|
const active = auth.accounts[auth.active];
|
|
|
|
client.user = client.users.get(active.session.user_id);
|
|
|
|
if (!navigator.onLine) {
|
|
|
|
return setStatus(ClientStatus.OFFLINE);
|
|
|
|
}
|
|
|
|
|
2021-06-19 13:46:05 -04:00
|
|
|
if (operations.ready())
|
2021-06-18 17:47:25 -04:00
|
|
|
setStatus(ClientStatus.CONNECTING);
|
|
|
|
|
|
|
|
if (navigator.onLine) {
|
|
|
|
await client
|
|
|
|
.fetchConfiguration()
|
|
|
|
.catch(() =>
|
|
|
|
console.error("Failed to connect to API server.")
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
await client.fetchConfiguration();
|
|
|
|
const callback = await client.useExistingSession(
|
|
|
|
active.session
|
|
|
|
);
|
|
|
|
|
2021-06-21 08:28:26 -04:00
|
|
|
if (callback) {
|
|
|
|
openScreen({ id: "onboarding", callback });
|
|
|
|
}
|
2021-06-18 17:47:25 -04:00
|
|
|
} catch (err) {
|
|
|
|
setStatus(ClientStatus.DISCONNECTED);
|
|
|
|
const error = takeError(err);
|
2021-06-21 08:28:26 -04:00
|
|
|
if (error === "Forbidden" || error === "Unauthorized") {
|
2021-06-19 13:46:05 -04:00
|
|
|
operations.logout(true);
|
2021-06-21 08:28:26 -04:00
|
|
|
openScreen({ id: "signed_out" });
|
2021-06-18 17:47:25 -04:00
|
|
|
} else {
|
2021-06-21 08:28:26 -04:00
|
|
|
openScreen({ id: "error", error });
|
2021-06-18 17:47:25 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2021-06-19 07:34:53 -04:00
|
|
|
try {
|
|
|
|
await client.fetchConfiguration()
|
|
|
|
} catch (err) {
|
|
|
|
console.error("Failed to connect to API server.");
|
|
|
|
}
|
|
|
|
|
2021-06-18 17:47:25 -04:00
|
|
|
setStatus(ClientStatus.READY);
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
if (status === ClientStatus.LOADING) {
|
2021-06-22 11:29:47 -04:00
|
|
|
return <Preloader type="spinner" />;
|
2021-06-18 17:47:25 -04:00
|
|
|
}
|
|
|
|
|
2021-06-18 15:07:26 -04:00
|
|
|
return (
|
2021-06-19 13:46:05 -04:00
|
|
|
<AppContext.Provider value={client}>
|
|
|
|
<StatusContext.Provider value={status}>
|
|
|
|
<OperationsContext.Provider value={operations}>
|
|
|
|
{ children }
|
|
|
|
</OperationsContext.Provider>
|
|
|
|
</StatusContext.Provider>
|
2021-06-18 15:07:26 -04:00
|
|
|
</AppContext.Provider>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export default connectState<{ children: Children }>(
|
|
|
|
Context,
|
|
|
|
state => {
|
|
|
|
return {
|
|
|
|
auth: state.auth,
|
|
|
|
sync: state.sync
|
|
|
|
};
|
|
|
|
},
|
|
|
|
true
|
|
|
|
);
|