aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--backend/src/routes/admin.ts10
-rw-r--r--frontend/src/App.tsx2
-rw-r--r--frontend/src/contexts/AuthContext.tsx6
-rw-r--r--frontend/src/pages/Admin.tsx205
-rw-r--r--frontend/src/utils/authApi.ts1
5 files changed, 217 insertions, 7 deletions
diff --git a/backend/src/routes/admin.ts b/backend/src/routes/admin.ts
index 1950715..63d6ccf 100644
--- a/backend/src/routes/admin.ts
+++ b/backend/src/routes/admin.ts
@@ -7,13 +7,9 @@ export const handleCreateGame = async (req: express.Request, res: express.Respon
return res.status(401).json({ error: 'Authentication required' });
}
- if (req.session.userId !== 1){
- return res.status(403).json({ error: 'Unauthorized. You are not the admin of this instance' });
- }
-
const user = await prisma.user.findUnique({
where: { id: req.session.userId },
- select: { id: true, username: true, email: true }
+ select: { id: true, username: true, isAdmin: true }
});
if (!user) {
@@ -22,6 +18,10 @@ export const handleCreateGame = async (req: express.Request, res: express.Respon
});
return res.status(401).json({ error: 'Invalid session' });
}
+
+ if (user.id !== 1 && !user.isAdmin) {
+ return res.status(403).json({ error: 'Unauthorized. You are not an admin of this instance' });
+ }
const { gameInternalName, gameFormattedName, gameDescription } = req.body;
if (!gameInternalName || !gameFormattedName || !gameDescription) {
diff --git a/frontend/src/App.tsx b/frontend/src/App.tsx
index 9c4e8bd..9b0e058 100644
--- a/frontend/src/App.tsx
+++ b/frontend/src/App.tsx
@@ -7,6 +7,7 @@ import Import from "./pages/Import";
import Home from "./pages/Home";
import Score from "./pages/Score";
import Chart from "./pages/Chart";
+import Admin from "./pages/Admin";
function App() {
return (
@@ -19,6 +20,7 @@ function App() {
<Route path="/home" element={<Home />} />
<Route path="/score" element={<Score />} />
<Route path="/chart" element={<Chart />} />
+ <Route path="/admin" element={<Admin />} />
</Routes>
</AuthProvider>
);
diff --git a/frontend/src/contexts/AuthContext.tsx b/frontend/src/contexts/AuthContext.tsx
index 8abeeb4..09e4864 100644
--- a/frontend/src/contexts/AuthContext.tsx
+++ b/frontend/src/contexts/AuthContext.tsx
@@ -4,9 +4,10 @@ import { authApi } from '../utils/authApi';
import type { User as ApiUser, SessionResponse } from '../utils/authApi';
interface User {
- id: string;
+ id: number;
username: string;
email: string;
+ isAdmin: boolean;
}
interface AuthContextType {
@@ -40,9 +41,10 @@ export const AuthProvider: React.FC<AuthProviderProps> = ({ children }) => {
const isAuthenticated = user !== null;
const transformApiUser = (apiUser: ApiUser): User => ({
- id: apiUser.id.toString(),
+ id: apiUser.id,
username: apiUser.username,
email: apiUser.email,
+ isAdmin: apiUser.isAdmin,
});
const checkAuth = async () => {
diff --git a/frontend/src/pages/Admin.tsx b/frontend/src/pages/Admin.tsx
new file mode 100644
index 0000000..f494fc2
--- /dev/null
+++ b/frontend/src/pages/Admin.tsx
@@ -0,0 +1,205 @@
+import { useNavigate } from "react-router";
+import { NavBar } from "../components/NavBar";
+import { useAuth } from "../contexts/AuthContext";
+import SessionExpiredPopup from "../components/SessionExpiredPopup";
+import { useState } from "react";
+
+
+const Admin = () => {
+ const { user, isLoading, logout } = useAuth();
+ const [showAddGame, setShowAddGame] = useState(false);
+ const [formData, setFormData] = useState({
+ gameInternalName: '',
+ gameFormattedName: '',
+ gameDescription: ''
+ });
+ const [isSubmitting, setIsSubmitting] = useState(false);
+ const navigate = useNavigate();
+
+ const handleLogout = async () => {
+ try {
+ await logout();
+ navigate("/");
+ } catch (error) {
+ console.error("Logout failed:", error);
+ alert("Network error during logout. Please try again.");
+ }
+ };
+
+ const handleInputChange = (e: React.ChangeEvent<HTMLInputElement>) => {
+ const { name, value } = e.target;
+ setFormData(prev => ({
+ ...prev,
+ [name]: value
+ }));
+ };
+
+ const handleSubmit = async (e: React.FormEvent) => {
+ e.preventDefault();
+
+ if (!formData.gameInternalName || !formData.gameFormattedName || !formData.gameDescription) {
+ alert('Please fill in all fields');
+ return;
+ }
+
+ setIsSubmitting(true);
+
+ try {
+ const response = await fetch(import.meta.env.VITE_API_URL + '/admin/createGame', {
+ method: 'POST',
+ headers: {
+ 'Content-Type': 'application/json',
+ },
+ credentials: 'include',
+ body: JSON.stringify(formData),
+ });
+
+ if (!response.ok) {
+ const error = await response.json();
+ throw new Error(error.error || 'Failed to create game');
+ }
+
+ alert('Game created successfully!');
+ setFormData({
+ gameInternalName: '',
+ gameFormattedName: '',
+ gameDescription: ''
+ });
+ setShowAddGame(false);
+
+ } catch (error) {
+ console.error('Failed to create game:', error);
+ alert(error instanceof Error ? error.message : 'Failed to create game');
+ } finally {
+ setIsSubmitting(false);
+ }
+ };
+
+ if (isLoading) {
+ return (
+ <div className="min-h-screen bg-slate-950 flex items-center justify-center">
+ <div className="text-center">
+ <div className="w-8 h-8 border-2 border-violet-500 border-t-transparent rounded-full animate-spin mx-auto mb-4"></div>
+ <p className="text-slate-300">Loading Admin dashboard...</p>
+ </div>
+ </div>
+ );
+ }
+
+ if (!user) {
+ return <SessionExpiredPopup />;
+ }
+ if(!user.isAdmin && user.id != 1){
+ console.log(user.id == 1)
+ return <div className="min-h-screen bg-slate-950 flex items-center justify-center">
+ <div className="text-center">
+ <div className="w-8 h-8 border-2 border-violet-500 border-t-transparent rounded-full animate-spin mx-auto mb-4"></div>
+ <p className="text-slate-400">You are not authorized to access this page.</p>
+ </div>
+ </div>;
+ }
+
+ return (
+ <div className="min-h-screen bg-slate-950">
+ <NavBar user={user} handleLogout={handleLogout} currentPage="home"/>
+
+ {/* Main Content */}
+ <div className="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
+ {/* Header */}
+ <div className="mb-8">
+ <h1 className="text-3xl font-bold text-white mb-2">Admin Page</h1>
+ <p className="text-slate-400">
+ Welcome Mirage Webmaster! Here are a variety of settings and tools you can use to customize the experience
+ </p>
+ </div>
+ {/* Add New Game Section */}
+ <div className="mb-8">
+ <div className="bg-slate-900 rounded-lg border border-slate-700">
+ <button
+ className="w-full px-6 py-4 text-left flex items-center justify-between hover:bg-slate-800 transition-colors rounded-lg"
+ onClick={() => setShowAddGame(!showAddGame)}
+ >
+ <h2 className="text-xl font-semibold text-white">Add New Game</h2>
+ <svg
+ className={`w-5 h-5 text-slate-400 transform transition-transform ${
+ showAddGame ? 'rotate-180' : ''
+ }`}
+ fill="none"
+ stroke="currentColor"
+ viewBox="0 0 24 24"
+ >
+ <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M19 9l-7 7-7-7" />
+ </svg>
+ </button>
+ {showAddGame && (
+ <div className="px-6 pb-6">
+ <p className="text-slate-300 leading-relaxed mb-6 p-4 bg-slate-800/50 rounded-md border-l-4 border-violet-500">
+ This form allows you to add a new game to Mirage. By default, Mirage will attempt to derive a method of showing the game's score on its own.
+ You may override this behavior by writing your own custom score display logic.
+ </p>
+ <form className="space-y-4" onSubmit={handleSubmit}>
+ <div>
+ <label htmlFor="gameInternalName" className="block text-sm font-medium text-slate-300 mb-2">
+ Game Internal Name
+ </label>
+ <input
+ type="text"
+ id="gameInternalName"
+ name="gameInternalName"
+ value={formData.gameInternalName}
+ onChange={handleInputChange}
+ className="w-full px-3 py-2 bg-slate-800 border border-slate-600 rounded-md text-white placeholder-slate-400 focus:outline-none focus:ring-2 focus:ring-violet-500 focus:border-transparent"
+ placeholder="The unique internal identifier for the game (i.e. dancerush)"
+ required
+ />
+ </div>
+ <div>
+ <label htmlFor="formattedName" className="block text-sm font-medium text-slate-300 mb-2">
+ Formatted Name
+ </label>
+ <input
+ type="text"
+ id="gameFormattedName"
+ name="gameFormattedName"
+ value={formData.gameFormattedName}
+ onChange={handleInputChange}
+ className="w-full px-3 py-2 bg-slate-800 border border-slate-600 rounded-md text-white placeholder-slate-400 focus:outline-none focus:ring-2 focus:ring-violet-500 focus:border-transparent"
+ placeholder="The formatted/stylized name users will see (i.e DANCERUSH STARDOM)"
+ required
+ />
+ </div>
+ <div>
+ <label htmlFor="formattedName" className="block text-sm font-medium text-slate-300 mb-2">
+ Game Description
+ </label>
+ <input
+ type="text"
+ id="gameDescription"
+ name="gameDescription"
+ value={formData.gameDescription}
+ onChange={handleInputChange}
+ className="w-full px-3 py-2 bg-slate-800 border border-slate-600 rounded-md text-white placeholder-slate-400 focus:outline-none focus:ring-2 focus:ring-violet-500 focus:border-transparent"
+ placeholder="A brief description of the game"
+ required
+ />
+ </div>
+ <div className="pt-4">
+ <button
+ type="submit"
+ disabled={isSubmitting}
+ className="bg-violet-600 hover:bg-violet-700 disabled:bg-violet-800 disabled:cursor-not-allowed text-white font-medium py-2 px-4 rounded-md transition-colors"
+ >
+ {isSubmitting ? 'Adding Game...' : 'Add Game'}
+ </button>
+ </div>
+ </form>
+ </div>
+ )}
+ </div>
+ </div>
+ </div>
+ </div>
+ );
+};
+
+export default Admin;
diff --git a/frontend/src/utils/authApi.ts b/frontend/src/utils/authApi.ts
index 469553d..1d6cc70 100644
--- a/frontend/src/utils/authApi.ts
+++ b/frontend/src/utils/authApi.ts
@@ -155,6 +155,7 @@ export interface User {
id: number;
username: string;
email: string;
+ isAdmin: boolean;
}
export interface SessionResponse {
send patches to the email below
yukais@pinapelz.com
include the subject [PATCH repo_name]
pinapelz.com
homepage