Refactor the API, then give the user a landing page that shows their profile #286
@ -1,6 +1,6 @@
|
||||
use axum::{
|
||||
extract::Path,
|
||||
http::{header::CONTENT_TYPE, HeaderMap, Method, StatusCode},
|
||||
http::{header::{AUTHORIZATION, CONTENT_TYPE}, HeaderMap, Method},
|
||||
routing::{get, post, put},
|
||||
Json, Router,
|
||||
};
|
||||
@ -8,7 +8,7 @@ use tower_http::cors::{Any, CorsLayer};
|
||||
|
||||
use crate::{
|
||||
core::Core,
|
||||
database::{SessionId, UserId},
|
||||
database::UserId,
|
||||
handlers::{
|
||||
check_password, create_game, create_user, get_user, healthcheck, set_password,
|
||||
wrap_handler, AuthRequest, CreateGameRequest, CreateUserRequest, SetPasswordRequest,
|
||||
@ -34,8 +34,12 @@ pub fn routes(core: Core) -> Router {
|
||||
post({
|
||||
let core = core.clone();
|
||||
move |req: Json<AuthRequest>| wrap_handler(|| check_password(core, req))
|
||||
}).layer(
|
||||
CorsLayer::new().allow_methods([Method::POST]).allow_headers([CONTENT_TYPE]).allow_origin(Any),
|
||||
})
|
||||
.layer(
|
||||
CorsLayer::new()
|
||||
.allow_methods([Method::POST])
|
||||
.allow_headers([CONTENT_TYPE])
|
||||
.allow_origin(Any),
|
||||
),
|
||||
)
|
||||
.route(
|
||||
@ -45,6 +49,12 @@ pub fn routes(core: Core) -> Router {
|
||||
let core = core.clone();
|
||||
move |headers: HeaderMap| wrap_handler(|| get_user(core, headers, None))
|
||||
})
|
||||
.layer(
|
||||
CorsLayer::new()
|
||||
.allow_methods([Method::GET])
|
||||
.allow_headers([AUTHORIZATION])
|
||||
.allow_origin(Any),
|
||||
)
|
||||
.put({
|
||||
let core = core.clone();
|
||||
move |headers: HeaderMap, req: Json<CreateUserRequest>| {
|
||||
|
@ -1,4 +1,4 @@
|
||||
import { SessionId } from "visions-types";
|
||||
import { SessionId, UserId, UserProfile } from "visions-types";
|
||||
|
||||
export type PlayingField = {
|
||||
backgroundImage: string;
|
||||
@ -70,11 +70,29 @@ export class Client {
|
||||
async auth(username: string, password: string): Promise<SessionId | undefined> {
|
||||
const url = new URL(this.base);
|
||||
url.pathname = `/api/v1/auth`
|
||||
const response = await fetch(url, { method: 'POST', headers: [['Content-Type', 'application/json']], body: JSON.stringify({ 'username': username, 'password': password }) });
|
||||
const response = await fetch(url, {
|
||||
method: 'POST',
|
||||
headers: [['Content-Type', 'application/json']],
|
||||
body: JSON.stringify({ 'username': username, 'password': password })
|
||||
});
|
||||
const session_id: SessionId = await response.json();
|
||||
return session_id;
|
||||
}
|
||||
|
||||
async profile(sessionId: SessionId, userId: UserId | undefined): Promise<UserProfile | undefined> {
|
||||
const url = new URL(this.base);
|
||||
if (userId) {
|
||||
url.pathname = `/api/v1/user${userId}`
|
||||
} else {
|
||||
url.pathname = `/api/v1/user`
|
||||
}
|
||||
const response = await fetch(url, {
|
||||
method: 'GET',
|
||||
headers: [['Authorization', `Bearer ${sessionId}`]],
|
||||
});
|
||||
return await response.json()
|
||||
}
|
||||
|
||||
async health() {
|
||||
const url = new URL(this.base);
|
||||
url.pathname = `/api/v1/health`;
|
||||
|
@ -1,10 +1,12 @@
|
||||
import React from 'react';
|
||||
import { Client } from '../../client';
|
||||
import { UserProfile } from 'visions-types';
|
||||
|
||||
interface ProfileProps {
|
||||
client: Client
|
||||
}
|
||||
export const ProfileElement = ({ name, games, is_admin }: UserProfile) => {
|
||||
const adminNote = is_admin ? <div> <i>Note: this user is an admin</i> </div> : <></>;
|
||||
|
||||
export const ProfileElement = ({ client }: ProfileProps) => {
|
||||
return <div></div>
|
||||
return (
|
||||
<div className="card">
|
||||
<h1>{name}</h1>
|
||||
<div>Games: {games.map((game) => <>{game.game_name} ({game.game_type})</>).join(', ')}</div>
|
||||
{adminNote}
|
||||
</div>)
|
||||
}
|
||||
|
@ -1,5 +1,6 @@
|
||||
import { ProfileElement } from './Profile/Profile'
|
||||
import { SimpleGuage } from './Guages/SimpleGuage'
|
||||
import { ThumbnailElement } from './Thumbnail/Thumbnail'
|
||||
import { TabletopElement } from './Tabletop/Tabletop'
|
||||
import { SimpleGuage } from './Guages/SimpleGuage'
|
||||
|
||||
export default { ThumbnailElement, TabletopElement, SimpleGuage }
|
||||
export { ProfileElement, ThumbnailElement, TabletopElement, SimpleGuage }
|
||||
|
@ -1,5 +1,7 @@
|
||||
import React, { useContext } from 'react';
|
||||
import { useContext, useEffect, useState } from 'react';
|
||||
import { UserProfile } from 'visions-types';
|
||||
import { Client } from '../../client';
|
||||
import { ProfileElement } from '../../components';
|
||||
import { getSessionId, StateContext, StateProvider } from '../../providers/StateProvider/StateProvider';
|
||||
|
||||
interface MainProps {
|
||||
@ -7,10 +9,21 @@ interface MainProps {
|
||||
}
|
||||
|
||||
export const MainView = ({ client }: MainProps) => {
|
||||
const [state, manager] = useContext(StateContext)
|
||||
const [state, _manager] = useContext(StateContext)
|
||||
const [profile, setProfile] = useState<UserProfile | undefined>(undefined)
|
||||
|
||||
const sessionId = getSessionId(state);
|
||||
const sessionId = getSessionId(state)
|
||||
useEffect(() => {
|
||||
if (sessionId) {
|
||||
client.profile(sessionId, undefined).then((profile) => setProfile(profile))
|
||||
}
|
||||
}, [sessionId, client])
|
||||
|
||||
return <div>Profile: {sessionId}</div>
|
||||
return (
|
||||
<div>
|
||||
<div>Session ID: {sessionId}</div>
|
||||
{profile && <ProfileElement {...profile} />}
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user