Move the assertNever utility to a utilities file

This commit is contained in:
Savanni D'Gerinel 2025-01-03 11:59:33 -05:00
parent dc8cb834e0
commit 08462388ea
16 changed files with 208 additions and 117 deletions

15
Cargo.lock generated
View File

@ -4394,6 +4394,20 @@ dependencies = [
"tracing", "tracing",
] ]
[[package]]
name = "tower-http"
version = "0.6.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "403fa3b783d4b626a8ad51d766ab03cb6d2dbfc46b1c5d4448395e6628dc9697"
dependencies = [
"bitflags 2.6.0",
"bytes",
"http 1.2.0",
"pin-project-lite",
"tower-layer",
"tower-service",
]
[[package]] [[package]]
name = "tower-layer" name = "tower-layer"
version = "0.3.3" version = "0.3.3"
@ -4688,6 +4702,7 @@ dependencies = [
"thiserror 2.0.3", "thiserror 2.0.3",
"tokio", "tokio",
"tokio-stream", "tokio-stream",
"tower-http",
"typeshare", "typeshare",
"urlencoding", "urlencoding",
"uuid 1.11.0", "uuid 1.11.0",

View File

@ -24,6 +24,7 @@ serde_json = { version = "*" }
thiserror = { version = "2.0.3" } thiserror = { version = "2.0.3" }
tokio = { version = "1", features = [ "full" ] } tokio = { version = "1", features = [ "full" ] }
tokio-stream = { version = "0.1.16" } tokio-stream = { version = "0.1.16" }
tower-http = { version = "0.6.2", features = ["cors"] }
typeshare = { version = "1.0.4" } typeshare = { version = "1.0.4" }
urlencoding = { version = "2.1.3" } urlencoding = { version = "2.1.3" }
uuid = { version = "1.11.0", features = ["v4"] } uuid = { version = "1.11.0", features = ["v4"] }

View File

@ -2,9 +2,11 @@ use std::fmt;
use rusqlite::types::{FromSql, FromSqlResult, ValueRef}; use rusqlite::types::{FromSql, FromSqlResult, ValueRef};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use typeshare::typeshare;
use uuid::Uuid; use uuid::Uuid;
#[derive(Clone, Debug, Deserialize, Eq, Hash, PartialEq, Serialize)] #[derive(Clone, Debug, Deserialize, Eq, Hash, PartialEq, Serialize)]
#[typeshare]
pub struct UserId(String); pub struct UserId(String);
impl UserId { impl UserId {
@ -47,6 +49,7 @@ impl fmt::Display for UserId {
} }
#[derive(Clone, Debug, Deserialize, Eq, Hash, PartialEq, Serialize)] #[derive(Clone, Debug, Deserialize, Eq, Hash, PartialEq, Serialize)]
#[typeshare]
pub struct SessionId(String); pub struct SessionId(String);
impl SessionId { impl SessionId {
@ -87,6 +90,7 @@ impl fmt::Display for SessionId {
} }
#[derive(Clone, Debug, Deserialize, Eq, Hash, PartialEq, Serialize)] #[derive(Clone, Debug, Deserialize, Eq, Hash, PartialEq, Serialize)]
#[typeshare]
pub struct GameId(String); pub struct GameId(String);
impl GameId { impl GameId {
@ -121,6 +125,7 @@ impl FromSql for GameId {
} }
#[derive(Clone, Debug, Deserialize, Eq, Hash, PartialEq, Serialize)] #[derive(Clone, Debug, Deserialize, Eq, Hash, PartialEq, Serialize)]
#[typeshare]
pub struct CharacterId(String); pub struct CharacterId(String);
impl CharacterId { impl CharacterId {

View File

@ -16,7 +16,7 @@ use crate::{
#[derive(Clone, Debug, PartialEq, Deserialize, Serialize)] #[derive(Clone, Debug, PartialEq, Deserialize, Serialize)]
pub struct HealthCheck { pub struct HealthCheck {
pub ok: bool, pub admin_enabled: bool,
} }
pub async fn wrap_handler<F, A, Fut>(f: F) -> (StatusCode, Json<Option<A>>) pub async fn wrap_handler<F, A, Fut>(f: F) -> (StatusCode, Json<Option<A>>)
@ -46,10 +46,10 @@ where
pub async fn healthcheck(core: Core) -> Vec<u8> { pub async fn healthcheck(core: Core) -> Vec<u8> {
match core.status().await { match core.status().await {
ResultExt::Ok(s) => serde_json::to_vec(&HealthCheck { ResultExt::Ok(s) => serde_json::to_vec(&HealthCheck {
ok: s.admin_enabled, admin_enabled: s.admin_enabled,
}) })
.unwrap(), .unwrap(),
ResultExt::Err(_) => serde_json::to_vec(&HealthCheck { ok: false }).unwrap(), ResultExt::Err(_) => serde_json::to_vec(&HealthCheck { admin_enabled: false }).unwrap(),
ResultExt::Fatal(err) => panic!("{}", err), ResultExt::Fatal(err) => panic!("{}", err),
} }
} }

View File

@ -1,9 +1,10 @@
use axum::{ use axum::{
extract::Path, extract::Path,
http::{HeaderMap, StatusCode}, http::{header::CONTENT_TYPE, HeaderMap, Method, StatusCode},
routing::{get, post, put}, routing::{get, post, put},
Json, Router, Json, Router,
}; };
use tower_http::cors::{Any, CorsLayer};
use crate::{ use crate::{
core::Core, core::Core,
@ -21,14 +22,21 @@ pub fn routes(core: Core) -> Router {
get({ get({
let core = core.clone(); let core = core.clone();
move || healthcheck(core) move || healthcheck(core)
}), })
.layer(
CorsLayer::new()
.allow_methods([Method::GET])
.allow_origin(Any),
),
) )
.route( .route(
"/api/v1/auth", "/api/v1/auth",
post({ post({
let core = core.clone(); let core = core.clone();
move |req: Json<AuthRequest>| wrap_handler(|| check_password(core, req)) move |req: Json<AuthRequest>| wrap_handler(|| check_password(core, req))
}), }).layer(
CorsLayer::new().allow_methods([Method::POST]).allow_headers([CONTENT_TYPE]).allow_origin(Any),
),
) )
.route( .route(
// By default, just get the self user. // By default, just get the self user.

View File

@ -46,9 +46,9 @@ const App = ({ client }: AppProps) => {
console.log("rendering app"); console.log("rendering app");
const [websocketUrl, setWebsocketUrl] = useState<string | undefined>(undefined); const [websocketUrl, setWebsocketUrl] = useState<string | undefined>(undefined);
useEffect(() => { // useEffect(() => {
client.registerWebsocket().then((url) => setWebsocketUrl(url)) // client.registerWebsocket().then((url) => setWebsocketUrl(url))
}, [client]); // }, [client]);
let router = let router =
createBrowserRouter([ createBrowserRouter([

View File

@ -1,3 +1,5 @@
import { SessionId } from "visions-types";
export type PlayingField = { export type PlayingField = {
backgroundImage: string; backgroundImage: string;
} }
@ -64,15 +66,15 @@ export class Client {
return fetch(url, { method: 'PUT', headers: [['Content-Type', 'application/json']], body: JSON.stringify(password) }); return fetch(url, { method: 'PUT', headers: [['Content-Type', 'application/json']], body: JSON.stringify(password) });
} }
async auth(username: string, password: string) { async auth(username: string, password: string): Promise<SessionId | undefined> {
const url = new URL(this.base); const url = new URL(this.base);
url.pathname = `api/v1/auth` url.pathname = `/api/v1/auth`
return fetch(url, { method: 'PUT', headers: [['Content-Type', 'application/json']], body: JSON.stringify({ 'username': username, 'password': password }) }); return fetch(url, { method: 'POST', headers: [['Content-Type', 'application/json']], body: JSON.stringify({ 'username': username, 'password': password }) });
} }
async status() { async health() {
const url = new URL(this.base); const url = new URL(this.base);
url.pathname = `/api/v1/status`; url.pathname = `/api/v1/health`;
return fetch(url).then((response) => response.json()); return fetch(url).then((response) => response.json());
} }

View File

@ -1,9 +1,9 @@
import React, { useContext } from 'react'; import React, { useContext } from 'react';
import './Tabletop.css'; import './Tabletop.css';
import { RGB } from 'visions-types'; import { Rgb } from 'visions-types';
interface TabletopElementProps { interface TabletopElementProps {
backgroundColor: RGB; backgroundColor: Rgb;
backgroundUrl: URL | undefined; backgroundUrl: URL | undefined;
} }

View File

@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { assertNever } from '.'; import { assertNever } from '../../utils';
import './Charsheet.css'; import './Charsheet.css';
import { DriveGuage } from './DriveGuage/DriveGuage'; import { DriveGuage } from './DriveGuage/DriveGuage';
import { Charsheet, Nerve, Cunning, Intuition } from './types'; import { Charsheet, Nerve, Cunning, Intuition } from './types';

View File

@ -1,9 +1,9 @@
import React from 'react'; import React from 'react';
import { assertNever } from '.';
import { SimpleGuage } from '../../components/Guages/SimpleGuage'; import { SimpleGuage } from '../../components/Guages/SimpleGuage';
import { Charsheet, Nerve, Cunning, Intuition } from './types'; import { Charsheet, Nerve, Cunning, Intuition } from './types';
import './CharsheetPanel.css'; import './CharsheetPanel.css';
import classNames from 'classnames'; import classNames from 'classnames';
import { assertNever } from '../../utils';
interface CharsheetPanelProps { interface CharsheetPanelProps {
sheet: Charsheet; sheet: Charsheet;

View File

@ -1,9 +1,5 @@
import { CharsheetElement } from './Charsheet'; import { CharsheetElement } from './Charsheet';
import { CharsheetPanelElement } from './CharsheetPanel'; import { CharsheetPanelElement } from './CharsheetPanel';
export function assertNever(value: never) {
throw new Error("Unexpected value: " + value);
}
export default { CharsheetElement, CharsheetPanelElement }; export default { CharsheetElement, CharsheetPanelElement };

View File

@ -1,26 +1,53 @@
import React, { createContext, PropsWithChildren, useCallback, useEffect, useReducer, useRef } from "react"; import React, { createContext, PropsWithChildren, useCallback, useEffect, useReducer, useRef } from "react";
import { Status, Tabletop } from "visions-types"; import { Status, Tabletop } from "visions-types";
import { Client } from "../../client"; import { Client } from "../../client";
import { assertNever } from "../../plugins/Candela"; import { assertNever } from "../../utils";
type AuthState = { type: "NoAdmin" } | { type: "Unauthed" } | { type: "Authed", userid: string }; type AuthState = { type: "NoAdmin" } | { type: "Unauthed" } | { type: "Authed", userid: string };
type AppState = { type LoadingState = { type: "Loading" }
auth: AuthState;
tabletop: Tabletop; type ReadyState = {
type: "Ready",
auth: AuthState,
tabletop: Tabletop,
} }
type AppState = LoadingState | ReadyState
type Action = { type: "SetAuthState", content: AuthState }; type Action = { type: "SetAuthState", content: AuthState };
/*
const initialState = (): AppState => ( const initialState = (): AppState => (
{ {
auth: { type: "NoAdmin" }, auth: { type: "NoAdmin" },
tabletop: { backgroundColor: { red: 0, green: 0, blue: 0 }, backgroundImage: undefined } tabletop: { backgroundColor: { red: 0, green: 0, blue: 0 }, backgroundImage: undefined }
} }
); );
*/
const initialState = (): AppState => ({ type: "Loading" })
const loadingReducer = (state: LoadingState, action: Action): AppState => {
return { type: "Ready", auth: action.content, tabletop: { backgroundColor: { red: 0, green: 0, blue: 0 }, backgroundImage: undefined } }
}
const readyReducer = (state: ReadyState, action: Action): AppState => {
return { ...state, auth: action.content }
}
const stateReducer = (state: AppState, action: Action): AppState => { const stateReducer = (state: AppState, action: Action): AppState => {
return { ...state, auth: action.content } switch (state.type) {
case "Loading": {
return loadingReducer(state, action);
}
case "Ready": {
return readyReducer(state, action);
}
default: {
assertNever(state);
return { type: "Loading" };
}
}
} }
class StateManager { class StateManager {
@ -35,7 +62,7 @@ class StateManager {
async status() { async status() {
if (!this.client || !this.dispatch) return; if (!this.client || !this.dispatch) return;
const { admin_enabled } = await this.client.status(); const { admin_enabled } = await this.client.health();
if (!admin_enabled) { if (!admin_enabled) {
this.dispatch({ type: "SetAuthState", content: { type: "NoAdmin" } }); this.dispatch({ type: "SetAuthState", content: { type: "NoAdmin" } });
} else { } else {
@ -54,9 +81,9 @@ class StateManager {
if (!this.client || !this.dispatch) return; if (!this.client || !this.dispatch) return;
let resp = await this.client.auth(username, password); let resp = await this.client.auth(username, password);
let userid = await resp.json(); let sessionid = await resp.json();
console.log("userid retrieved", userid); console.log("sessionid retrieved", sessionid);
this.dispatch({ type: "SetAuthState", content: { type: "Authed", userid } }); this.dispatch({ type: "SetAuthState", content: { type: "Authed", sessionid } });
} }
} }

4
visions/ui/src/utils.ts Normal file
View File

@ -0,0 +1,4 @@
export function assertNever(value: never) {
throw new Error("Unexpected value: " + value);
}

View File

@ -1,6 +1,6 @@
import { PropsWithChildren, useContext, useState } from 'react'; import { PropsWithChildren, useContext, useState } from 'react';
import { StateContext } from '../../providers/StateProvider/StateProvider'; import { StateContext } from '../../providers/StateProvider/StateProvider';
import { assertNever } from '../../plugins/Candela'; import { assertNever } from '../../utils';
import './Authentication.css'; import './Authentication.css';
interface AuthenticationProps { interface AuthenticationProps {
@ -17,6 +17,11 @@ export const Authentication = ({ onAdminPassword, onAuth, children }: PropsWithC
let [pwField, setPwField] = useState<string>(""); let [pwField, setPwField] = useState<string>("");
let [state, _] = useContext(StateContext); let [state, _] = useContext(StateContext);
switch (state.type) {
case "Loading": {
return <div>Loading</div>
}
case "Ready": {
switch (state.auth.type) { switch (state.auth.type) {
case "NoAdmin": { case "NoAdmin": {
return <div className="auth"> return <div className="auth">
@ -48,4 +53,12 @@ export const Authentication = ({ onAdminPassword, onAuth, children }: PropsWithC
return <div></div>; return <div></div>;
} }
} }
}
default: {
assertNever(state);
return <div></div>
}
}
} }

View File

@ -5,6 +5,7 @@ import { TabletopElement } from '../../components/Tabletop/Tabletop';
import { ThumbnailElement } from '../../components/Thumbnail/Thumbnail'; import { ThumbnailElement } from '../../components/Thumbnail/Thumbnail';
import { WebsocketContext } from '../../components/WebsocketProvider'; import { WebsocketContext } from '../../components/WebsocketProvider';
import './GmView.css'; import './GmView.css';
import { assertNever } from '../../utils';
interface GmViewProps { interface GmViewProps {
client: Client client: Client
@ -18,6 +19,9 @@ export const GmView = ({ client }: GmViewProps) => {
client.availableImages().then((images) => setImages(images)); client.availableImages().then((images) => setImages(images));
}, [client]); }, [client]);
switch (state.type) {
case "Loading": return <div></div>;
case "Ready": {
const backgroundUrl = state.tabletop.backgroundImage ? client.imageUrl(state.tabletop.backgroundImage) : undefined; const backgroundUrl = state.tabletop.backgroundImage ? client.imageUrl(state.tabletop.backgroundImage) : undefined;
return (<div className="gm-view"> return (<div className="gm-view">
<div> <div>
@ -25,5 +29,11 @@ export const GmView = ({ client }: GmViewProps) => {
</div> </div>
<TabletopElement backgroundColor={state.tabletop.backgroundColor} backgroundUrl={backgroundUrl} /> <TabletopElement backgroundColor={state.tabletop.backgroundColor} backgroundUrl={backgroundUrl} />
</div>) </div>)
}
default: {
assertNever(state);
return <div></div>;
}
}
} }

View File

@ -5,6 +5,7 @@ import { Client } from '../../client';
import { TabletopElement } from '../../components/Tabletop/Tabletop'; import { TabletopElement } from '../../components/Tabletop/Tabletop';
import Candela from '../../plugins/Candela'; import Candela from '../../plugins/Candela';
import { StateContext } from '../../providers/StateProvider/StateProvider'; import { StateContext } from '../../providers/StateProvider/StateProvider';
import { assertNever } from '../../utils';
const TEST_CHARSHEET_UUID = "12df9c09-1f2f-4147-8eda-a97bd2a7a803"; const TEST_CHARSHEET_UUID = "12df9c09-1f2f-4147-8eda-a97bd2a7a803";
@ -26,6 +27,9 @@ export const PlayerView = ({ client }: PlayerViewProps) => {
[client, setCharsheet] [client, setCharsheet]
); );
switch (state.type) {
case "Loading": return <div></div>;
case "Ready": {
const backgroundColor = state.tabletop.backgroundColor; const backgroundColor = state.tabletop.backgroundColor;
const tabletopColorStyle = `rgb(${backgroundColor.red}, ${backgroundColor.green}, ${backgroundColor.blue})`; const tabletopColorStyle = `rgb(${backgroundColor.red}, ${backgroundColor.green}, ${backgroundColor.blue})`;
const backgroundUrl = state.tabletop.backgroundImage ? client.imageUrl(state.tabletop.backgroundImage) : undefined; const backgroundUrl = state.tabletop.backgroundImage ? client.imageUrl(state.tabletop.backgroundImage) : undefined;
@ -35,5 +39,11 @@ export const PlayerView = ({ client }: PlayerViewProps) => {
<div className="player-view__right-panel"> <div className="player-view__right-panel">
{charsheet ? <Candela.CharsheetPanelElement sheet={charsheet} /> : <div> </div>}</div> {charsheet ? <Candela.CharsheetPanelElement sheet={charsheet} /> : <div> </div>}</div>
</div>) </div>)
}
default: {
assertNever(state);
return <div></div>;
}
}
} }