diff options
| -rw-r--r-- | backend/schema.prisma | 6 | ||||
| -rw-r--r-- | backend/src/index.ts | 3 | ||||
| -rw-r--r-- | backend/src/routes/auth.ts | 22 | ||||
| -rw-r--r-- | backend/src/routes/server.ts | 36 | ||||
| -rw-r--r-- | frontend/src/pages/Admin.tsx | 164 | ||||
| -rw-r--r-- | frontend/src/pages/Register.tsx | 50 |
6 files changed, 276 insertions, 5 deletions
diff --git a/backend/schema.prisma b/backend/schema.prisma index fbc2f90..26f57dd 100644 --- a/backend/schema.prisma +++ b/backend/schema.prisma @@ -55,3 +55,9 @@ model Charts { game Game @relation(fields: [gameInternalName], references: [internalName]) scores Score[] } + +model InviteCodes { + id Int @id @default(autoincrement()) + code String @unique + remaining Int +} diff --git a/backend/src/index.ts b/backend/src/index.ts index c0089c2..9511f28 100644 --- a/backend/src/index.ts +++ b/backend/src/index.ts @@ -11,6 +11,7 @@ import * as userRoutes from './routes/user'; import * as gameRoutes from './routes/game'; import * as scoreRoutes from './routes/score'; import * as adminRoutes from './routes/admin'; +import * as serverRoutes from './routes/server'; const app = express(); const port = 5000; @@ -48,6 +49,8 @@ startSessionCleanup(); app.post('/api/register', authRoutes.handleRegistration); app.post('/api/authenticate', authRoutes.handleAuthentication); app.post('/api/logout', requireAuth, authRoutes.handleLogout); +app.get('/api/info', serverRoutes.handleGetInstanceInfo); +app.post('/api/admin/createInvite', serverRoutes.handleCreateInviteCode); app.get('/api/me', userRoutes.handleMeRoute); app.get('/api/session', userRoutes.handleGetCurrentSession); diff --git a/backend/src/routes/auth.ts b/backend/src/routes/auth.ts index f857dea..8bc6274 100644 --- a/backend/src/routes/auth.ts +++ b/backend/src/routes/auth.ts @@ -6,12 +6,24 @@ import crypto from 'crypto'; export const handleRegistration = async (req: express.Request, res: express.Response) => { try { - const { username, password, email } = req.body; + const { username, password, email, code: inviteCode } = req.body; + const requireInvite = process.env.REQUIRE_INVITE === 'true'; if (!username || !password || !email) { return res.status(400).json({ error: 'All fields are required' }); } + if (requireInvite && !inviteCode) { + return res.status(400).json({ error: 'Invite code is required' }); + } + + if (requireInvite && inviteCode) { + const invite = await prisma.inviteCodes.findUnique({ where: { code: inviteCode } }); + if (!invite || invite.remaining <= 0) { + return res.status(400).json({ error: 'Invalid invite code' }); + } + } + const existingUser = await prisma.user.findFirst({ where: { OR: [ @@ -38,6 +50,14 @@ export const handleRegistration = async (req: express.Request, res: express.Resp } }); + // Decrement invite code usage if required + if (requireInvite && inviteCode) { + await prisma.inviteCodes.update({ + where: { code: inviteCode }, + data: { remaining: { decrement: 1 } } + }); + } + // Create session for the new user req.session.userId = user.id; const sessionId = await createSession(user.id); diff --git a/backend/src/routes/server.ts b/backend/src/routes/server.ts new file mode 100644 index 0000000..7377fff --- /dev/null +++ b/backend/src/routes/server.ts @@ -0,0 +1,36 @@ +import { prisma } from '../config/db'; +import express from 'express'; + +export const handleGetInstanceInfo = async (req: express.Request, res: express.Response) => { + try { + const userCount = await prisma.user.count(); + const requireInvite = process.env.REQUIRE_INVITE || false; + return res.status(200).json({ userCount, requireInvite }); + } catch (error) { + console.error('Unable to get instance info:', error); + res.status(500).json({ error: 'Internal server error' }); + } +} + +export const handleCreateInviteCode = async (req: express.Request, res: express.Response) => { + try { + const { uses, code } = req.body; + if (!uses) { + return res.status(400).json({ error: 'Missing required parameter: uses (number of maximum usages of this code)' }); + } + const codeAlreadyExists = await prisma.inviteCodes.findUnique({ where: { code } }); + if (codeAlreadyExists) { + return res.status(400).json({ error: 'Invite code already exists' }); + } + const inviteCode = await prisma.inviteCodes.create({ + data: { + code: code || Math.random().toString(36).substring(2, 15), + remaining: uses, + }, + }); + return res.status(200).json({ inviteCode }); + } catch (error) { + console.error('Unable to create invite code:', error); + res.status(500).json({ error: 'Internal server error' }); + } +} diff --git a/frontend/src/pages/Admin.tsx b/frontend/src/pages/Admin.tsx index f494fc2..043a32e 100644 --- a/frontend/src/pages/Admin.tsx +++ b/frontend/src/pages/Admin.tsx @@ -8,12 +8,19 @@ import { useState } from "react"; const Admin = () => { const { user, isLoading, logout } = useAuth(); const [showAddGame, setShowAddGame] = useState(false); + const [showCreateInvite, setShowCreateInvite] = useState(false); const [formData, setFormData] = useState({ gameInternalName: '', gameFormattedName: '', gameDescription: '' }); + const [inviteFormData, setInviteFormData] = useState({ + uses: '', + code: '' + }); const [isSubmitting, setIsSubmitting] = useState(false); + const [isCreatingInvite, setIsCreatingInvite] = useState(false); + const [createdInviteCode, setCreatedInviteCode] = useState<string | null>(null); const navigate = useNavigate(); const handleLogout = async () => { @@ -34,6 +41,14 @@ const Admin = () => { })); }; + const handleInviteInputChange = (e: React.ChangeEvent<HTMLInputElement>) => { + const { name, value } = e.target; + setInviteFormData(prev => ({ + ...prev, + [name]: value + })); + }; + const handleSubmit = async (e: React.FormEvent) => { e.preventDefault(); @@ -75,6 +90,65 @@ const Admin = () => { } }; + const handleInviteSubmit = async (e: React.FormEvent) => { + e.preventDefault(); + + if (!inviteFormData.uses) { + alert('Please specify the number of uses for the invite code'); + return; + } + + const uses = parseInt(inviteFormData.uses); + if (isNaN(uses) || uses <= 0) { + alert('Please enter a valid number of uses'); + return; + } + + setIsCreatingInvite(true); + + try { + const requestBody: { uses: number; code?: string } = { uses }; + if (inviteFormData.code.trim()) { + requestBody.code = inviteFormData.code.trim(); + } + + const response = await fetch(import.meta.env.VITE_API_URL + '/admin/createInvite', { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + }, + credentials: 'include', + body: JSON.stringify(requestBody), + }); + + if (!response.ok) { + const error = await response.json(); + throw new Error(error.error || 'Failed to create invite code'); + } + + const result = await response.json(); + setCreatedInviteCode(result.inviteCode.code); + setInviteFormData({ + uses: '', + code: '' + }); + + } catch (error) { + console.error('Failed to create invite code:', error); + alert(error instanceof Error ? error.message : 'Failed to create invite code'); + } finally { + setIsCreatingInvite(false); + } + }; + + const copyToClipboard = (text: string) => { + navigator.clipboard.writeText(text).then(() => { + alert('Invite code copied to clipboard!'); + }).catch(() => { + alert('Failed to copy to clipboard'); + }); + }; + if (isLoading) { return ( <div className="min-h-screen bg-slate-950 flex items-center justify-center"> @@ -112,6 +186,96 @@ const Admin = () => { Welcome Mirage Webmaster! Here are a variety of settings and tools you can use to customize the experience </p> </div> + + {/* Create Invite Code 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={() => setShowCreateInvite(!showCreateInvite)} + > + <h2 className="text-xl font-semibold text-white">Create Invite Code</h2> + <svg + className={`w-5 h-5 text-slate-400 transform transition-transform ${ + showCreateInvite ? '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> + {showCreateInvite && ( + <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"> + Generate invite codes to allow new users to register. You can specify how many times the code can be used + and optionally set a custom code (otherwise one will be generated automatically). + </p> + + {createdInviteCode && ( + <div className="mb-6 p-4 bg-green-900/30 border border-green-700 rounded-md"> + <h3 className="text-green-400 font-semibold mb-2">Invite Code Created Successfully!</h3> + <div className="flex items-center gap-2"> + <code className="bg-slate-800 px-3 py-2 rounded text-green-300 font-mono"> + {createdInviteCode} + </code> + <button + onClick={() => copyToClipboard(createdInviteCode)} + className="bg-green-600 hover:bg-green-700 text-white px-3 py-2 rounded-md text-sm transition-colors" + > + Copy + </button> + </div> + </div> + )} + + <form className="space-y-4" onSubmit={handleInviteSubmit}> + <div> + <label htmlFor="uses" className="block text-sm font-medium text-slate-300 mb-2"> + Number of Uses + </label> + <input + type="number" + id="uses" + name="uses" + value={inviteFormData.uses} + onChange={handleInviteInputChange} + 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="How many times this code can be used" + min="1" + required + /> + </div> + <div> + <label htmlFor="code" className="block text-sm font-medium text-slate-300 mb-2"> + Custom Code (Optional) + </label> + <input + type="text" + id="code" + name="code" + value={inviteFormData.code} + onChange={handleInviteInputChange} + 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="Leave blank to generate automatically" + /> + </div> + <div className="pt-4"> + <button + type="submit" + disabled={isCreatingInvite} + 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" + > + {isCreatingInvite ? 'Creating Invite Code...' : 'Create Invite Code'} + </button> + </div> + </form> + </div> + )} + </div> + </div> + {/* Add New Game Section */} <div className="mb-8"> <div className="bg-slate-900 rounded-lg border border-slate-700"> diff --git a/frontend/src/pages/Register.tsx b/frontend/src/pages/Register.tsx index 90edbfd..5b53f46 100644 --- a/frontend/src/pages/Register.tsx +++ b/frontend/src/pages/Register.tsx @@ -1,4 +1,4 @@ -import { useState } from 'react'; +import { useEffect, useState } from 'react'; import { Link, useNavigate } from 'react-router'; import { useAuth } from '../contexts/AuthContext'; @@ -7,11 +7,26 @@ const Register = () => { username: '', email: '', password: '', - confirmPassword: '' + confirmPassword: '', + code: '' }); const [errors, setErrors] = useState<Record<string, string>>({}); + const [requireInvite, setRequireInvite] = useState(false); const [isLoading, setIsLoading] = useState(false); + useEffect(() => { + const fetchServerInfo = async () => { + try { + const response = await fetch(import.meta.env.VITE_API_URL + "/info"); + const data = await response.json(); + setRequireInvite(Boolean(data.requireInvite)); + } catch (error) { + console.error('Error fetching server info:', error); + } + }; + fetchServerInfo(); + }, []); + const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => { const { name, value } = e.target; setFormData(prev => ({ @@ -53,6 +68,10 @@ const Register = () => { newErrors.confirmPassword = 'Passwords do not match'; } + if (requireInvite && !formData.code.trim()) { + newErrors.code = 'Invite code is required'; + } + setErrors(newErrors); return Object.keys(newErrors).length === 0; }; @@ -67,11 +86,14 @@ const Register = () => { setIsLoading(true); try { - const result = await register({ + const registrationData = { username: formData.username, email: formData.email, password: formData.password, - }); + ...(requireInvite && { code: formData.code }) + }; + + const result = await register(registrationData); if (!result.success) { setErrors({ general: result.error || 'Registration failed. Please try again.' }); @@ -131,6 +153,26 @@ const Register = () => { )} <form onSubmit={handleSubmit} className="space-y-6"> + {requireInvite && ( + <div> + <label htmlFor="code" className="block text-sm font-medium text-slate-300 mb-2"> + Invite code + </label> + <input + type="text" + id="code" + name="code" + value={formData.code} + onChange={handleChange} + className="block w-full px-3 py-2 border border-slate-600 rounded-md shadow-sm bg-slate-800 text-white placeholder-slate-400 focus:outline-none focus:ring-2 focus:ring-violet-500 focus:border-violet-500 transition-colors" + placeholder="Enter your invite code" + /> + {errors.code && ( + <p className="mt-1 text-sm text-red-400">{errors.code}</p> + )} + </div> + )} + <div> <label htmlFor="username" className="block text-sm font-medium text-slate-300 mb-2"> Username |
