Compare commits

...

4 Commits

Author SHA1 Message Date
6ef829dbc8 Authorize join lobby 2024-07-09 18:38:13 -05:00
800fa62ec6 Authenticate with the server 2024-05-25 16:26:29 -05:00
6b45296293 send auth message to server 2024-05-25 14:57:33 -05:00
85c0e27a32 Create functions for websockets 2024-05-25 14:31:52 -05:00
3 changed files with 115 additions and 23 deletions

6
src/assets/ok.svg Normal file
View File

@ -0,0 +1,6 @@
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 100 100" preserveAspectRatio="xMidYMid"
style="shape-rendering: auto; display: block;" width="50" height="50">
<circle stroke-linecap="round" fill="none" stroke="#16a34a"
stroke-width="8" r="32" cy="50" cx="50" data-idx="2">
</circle>
</svg>

After

Width:  |  Height:  |  Size: 314 B

View File

@ -4,8 +4,8 @@ import { backend, set_auth_token, UserInfo } from "../utils";
import { Card } from "../components/Card"; import { Card } from "../components/Card";
import { AxiosError } from "axios"; import { AxiosError } from "axios";
const userIdKey = "UserId"; export const userIdKey = "UserId";
const usernameKey = "Username"; export const usernameKey = "Username";
export function Index() { export function Index() {
const [userInfo, setUserInfo] = createSignal<UserInfo | null>(null); const [userInfo, setUserInfo] = createSignal<UserInfo | null>(null);
@ -66,7 +66,7 @@ export function Index() {
<p>Validating user id...</p> <p>Validating user id...</p>
</Show> </Show>
<Show when={userInfo() !== null && validated()}> <Show when={userInfo() !== null && validated()}>
<LobbyConnection /> <LobbyConnection user_id={userInfo()!.UserId} />
</Show> </Show>
</div> </div>
<hr /> <hr />
@ -162,7 +162,7 @@ function UserRegistration(props: {setUserInfo: (u: UserInfo) => void}) {
); );
} }
function LobbyConnection() { function LobbyConnection(props: {user_id: string}) {
const [loading, setLoading] = createSignal(false); const [loading, setLoading] = createSignal(false);
const [error, setError] = createSignal(""); const [error, setError] = createSignal("");
const navigate = useNavigate(); const navigate = useNavigate();
@ -175,7 +175,9 @@ function LobbyConnection() {
setLoading(true); setLoading(true);
try { try {
const res = await backend.post<{LobbyId: string}>("/lobby/new"); const res = await backend.post<{LobbyId: string}>("/lobby/new", {
UserId: props.user_id,
});
console.log(res.data); console.log(res.data);
// Redirect to the lobby component using the received lobby id // Redirect to the lobby component using the received lobby id
navigate(`/lobby/${res.data.LobbyId}`); navigate(`/lobby/${res.data.LobbyId}`);

View File

@ -1,39 +1,102 @@
import { Show, createSignal, onMount } from "solid-js"; import { Show, createSignal, onCleanup, onMount } from "solid-js";
import LoadingIcon from "../assets/loading.svg"; import LoadingIcon from "../assets/loading.svg";
import ErrorIcon from "../assets/error.svg"; import ErrorIcon from "../assets/error.svg";
import ConnectedIcon from "../assets/ok.svg";
import { userIdKey } from "./Index";
import { useNavigate, useParams } from "@solidjs/router";
enum LobbyStatus { enum LobbyStatus {
Connecting, Connecting,
Connected, Connected,
Disconnected, Disconnected,
Authenticating,
Error, Error,
} }
type LobbyMsg = {
action: string,
value: string,
}
const connectionRetryInterval = 5000;
export function Lobby() { export function Lobby() {
const [status, setStatus] = createSignal(LobbyStatus.Connecting); const [status, setStatus] = createSignal(LobbyStatus.Disconnected);
const navigate = useNavigate();
const params = useParams();
let ws: WebSocket|null = null; let ws: WebSocket|null = null;
const lobbyConnect = async() => { const lobbyConnect = () => {
setStatus(LobbyStatus.Connecting);
ws = new WebSocket(`${import.meta.env.VITE_WS_URL}/lobby/connect`); ws = new WebSocket(`${import.meta.env.VITE_WS_URL}/lobby/connect`);
ws.addEventListener("open", () => { ws.addEventListener("open", onWsOpen);
console.log("Connection open!"); ws.addEventListener("message", onWsMessage);
setStatus(LobbyStatus.Connected); ws.addEventListener("error", onWsError);
}); ws.addEventListener("close", onWsClose);
};
ws.addEventListener("message", (ev) => { const onWsOpen = () => {
console.log("message from ws!"); console.log("Connection open. Authenticating...");
console.log(ev); setStatus(LobbyStatus.Authenticating);
});
ws.addEventListener("error", (ev) => { // The first message must be authenticating with the server
console.error(ev); // TODO
setStatus(LobbyStatus.Error);
});
ws.addEventListener("close", () => { const userId = localStorage.getItem(userIdKey);
console.log("connection closed"); if (userId === null) {
}); // Redirect to home page
ws?.close(1000);
navigate("/");
return;
}
// Send an auth message to the server
// The server will respond with either "authenticated" or "unauthenticated"
// The message sent is <userid>,<lobbyid>
ws!.send(JSON.stringify({
action: "auth",
value: `${userId},${params.id}`,
}));
};
const onWsMessage = (ev: MessageEvent) => {
const data = JSON.parse(ev.data) as LobbyMsg;
switch (data.action) {
case "auth": {
if (data.value === "authenticated") {
setStatus(LobbyStatus.Connected);
} else if (data.value === "unauthenticated") {
ws?.close(1000);
navigate("/");
}
break;
}
}
};
const onWsError = (ev: Event) => {
console.error("error in connection");
console.error(ev);
setStatus(LobbyStatus.Error);
};
const onWsClose = (ev: CloseEvent) => {
if (ev.code === 1000) {
// The connection was normally closed
return;
}
// If the previous state is "Error", we don't set
// the state to Disconnected
if (status() !== LobbyStatus.Error) {
setStatus(LobbyStatus.Disconnected);
}
console.log(`connection closed. reconnecting in ${connectionRetryInterval}ms`);
setTimeout(lobbyConnect, connectionRetryInterval);
}; };
const send = () => { const send = () => {
@ -47,6 +110,10 @@ export function Lobby() {
}; };
onMount(lobbyConnect); onMount(lobbyConnect);
onCleanup(() => {
console.log("lobby cleanup");
ws?.close(1000);
});
return ( return (
<div class="w-[40rem] mx-auto bg-c-bg-2 my-4 p-4 rounded shadow-md"> <div class="w-[40rem] mx-auto bg-c-bg-2 my-4 p-4 rounded shadow-md">
@ -54,12 +121,24 @@ export function Lobby() {
Lobby Lobby
</h1> </h1>
<Show when={status() === LobbyStatus.Disconnected}>
<p>
<img class="inline-block h-6 animate-spin" src={LoadingIcon} alt="..." />
Disconected. Attempting reconnection in {connectionRetryInterval / 1000} seconds.
</p>
</Show>
<Show when={status() === LobbyStatus.Connecting}> <Show when={status() === LobbyStatus.Connecting}>
<p> <p>
<img class="inline-block h-6 animate-spin" src={LoadingIcon} alt="..." /> <img class="inline-block h-6 animate-spin" src={LoadingIcon} alt="..." />
Connecting to the lobby Connecting to the lobby
</p> </p>
</Show> </Show>
<Show when={status() === LobbyStatus.Authenticating}>
<p>
<img class="inline-block h-6 animate-spin" src={LoadingIcon} alt="..." />
Authenticating with the server
</p>
</Show>
<Show when={status() === LobbyStatus.Error}> <Show when={status() === LobbyStatus.Error}>
<p> <p>
@ -69,6 +148,11 @@ export function Lobby() {
</Show> </Show>
<Show when={status() === LobbyStatus.Connected}> <Show when={status() === LobbyStatus.Connected}>
<p>
<img class="inline-block h-6" src={ConnectedIcon} alt="..." />
Connected
</p>
<button class="bg-cyan-500 p-2 rounded text-white" onClick={send}>Send message</button> <button class="bg-cyan-500 p-2 rounded text-white" onClick={send}>Send message</button>
</Show> </Show>