From 9adc0fe19b7644faff0fc5a198e82ab7565db0e4 Mon Sep 17 00:00:00 2001 From: caadiq Date: Mon, 22 Dec 2025 11:42:37 +0900 Subject: [PATCH] =?UTF-8?q?=ED=94=84=EB=A1=9C=ED=95=84=20=EC=86=8C?= =?UTF-8?q?=EC=BC=93=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- backend/routes/auth.js | 31 +++++++ backend/routes/link.js | 101 +++++++++++++++++++++- frontend/src/components/Sidebar.jsx | 36 +++++++- frontend/src/pages/Admin.jsx | 46 ++++++++-- frontend/src/pages/LoginPage.jsx | 8 +- frontend/src/pages/PlayerStatsPage.jsx | 21 +++-- frontend/src/pages/PlayersPage.jsx | 21 +++-- frontend/src/pages/ProfilePage.jsx | 114 ++++++++++++++++++++++++- frontend/src/pages/ServerDetail.jsx | 42 ++++++--- frontend/src/pages/WorldsPage.jsx | 39 +++++++-- 10 files changed, 414 insertions(+), 45 deletions(-) diff --git a/backend/routes/auth.js b/backend/routes/auth.js index e79d9ed..5bb5ba6 100644 --- a/backend/routes/auth.js +++ b/backend/routes/auth.js @@ -363,4 +363,35 @@ router.post("/logout", (req, res) => { res.json({ success: true }); }); +/** + * DELETE /auth/delete - 회원 탈퇴 + */ +router.delete("/delete", async (req, res) => { + const authHeader = req.headers.authorization; + if (!authHeader || !authHeader.startsWith("Bearer ")) { + return res.status(401).json({ error: "로그인이 필요합니다." }); + } + + const token = authHeader.split(" ")[1]; + + try { + const decoded = jwt.verify(token, JWT_SECRET); + + // 연동 정보 삭제 + await pool.query("DELETE FROM minecraft_links WHERE user_id = ?", [ + decoded.id, + ]); + + // 사용자 삭제 + await pool.query("DELETE FROM users WHERE id = ?", [decoded.id]); + + console.log(`[Auth] 회원 탈퇴: id=${decoded.id}, email=${decoded.email}`); + + res.json({ success: true, message: "회원 탈퇴가 완료되었습니다." }); + } catch (error) { + console.error("[Auth] 회원 탈퇴 오류:", error); + res.status(500).json({ error: "서버 오류가 발생했습니다." }); + } +}); + export default router; diff --git a/backend/routes/link.js b/backend/routes/link.js index b46bb0b..48b55ec 100644 --- a/backend/routes/link.js +++ b/backend/routes/link.js @@ -209,10 +209,41 @@ router.get("/status", async (req, res) => { return res.json({ linked: false }); } + const uuid = links[0].minecraft_uuid; + let currentName = links[0].minecraft_name; + + // 모드 API에서 최신 닉네임 조회 및 동기화 + try { + const MOD_API_URL = + process.env.MOD_API_URL || "http://minecraft-server:8080"; + const modRes = await fetch(`${MOD_API_URL}/player/${uuid}`); + if (modRes.ok) { + const playerData = await modRes.json(); + if (playerData.name && playerData.name !== currentName) { + // 닉네임이 변경되었으면 minecraft_links 및 users 테이블 모두 업데이트 + await pool.query( + "UPDATE minecraft_links SET minecraft_name = ? WHERE user_id = ?", + [playerData.name, user.id] + ); + await pool.query("UPDATE users SET name = ? WHERE id = ?", [ + playerData.name, + user.id, + ]); + currentName = playerData.name; + console.log( + `[Link] 닉네임 동기화: ${links[0].minecraft_name} → ${currentName}` + ); + } + } + } catch (modErr) { + // 모드 API 호출 실패해도 기존 데이터로 응답 + console.log("[Link] 닉네임 동기화 실패 (모드 API 오류):", modErr.message); + } + res.json({ linked: true, - minecraftName: links[0].minecraft_name, - minecraftUuid: links[0].minecraft_uuid, + minecraftName: currentName, + minecraftUuid: uuid, linkedAt: links[0].linked_at, }); } catch (error) { @@ -384,4 +415,70 @@ router.post("/cancel", express.text({ type: "*/*" }), async (req, res) => { } }); +/** + * GET /link/skin/:type/:uuid/:size - 스킨 URL 조회 (캐싱) + * type: avatar 또는 body + * RustFS에 있으면 S3 URL 반환, 없으면 mc-heads에서 다운로드하여 저장 후 반환 + */ +router.get("/skin/:type/:uuid/:size", async (req, res) => { + const { type, uuid, size } = req.params; + + if (!uuid || uuid.length < 32) { + return res.status(400).json({ error: "유효하지 않은 UUID입니다." }); + } + + const validTypes = ["avatar", "body"]; + const skinType = validTypes.includes(type) ? type : "avatar"; + const skinSize = parseInt(size) || 128; + + // S3 URL (타입별로 폴더 분리) + const s3Key = `skins/${skinType}/${uuid}_${skinSize}.png`; + const s3Url = `https://s3.caadiq.co.kr/minecraft/${s3Key}`; + + try { + // S3에 파일 있는지 HEAD 요청으로 확인 + const headRes = await fetch(s3Url, { method: "HEAD" }); + + if (headRes.ok) { + // 이미 캐시됨 + return res.json({ url: s3Url, cached: true }); + } + + // 없으면 mc-heads에서 다운로드 + console.log(`[Link] 스킨 캐싱: ${skinType}/${uuid}/${skinSize}`); + const skinUrl = `https://mc-heads.net/${skinType}/${uuid}/${skinSize}`; + const skinRes = await fetch(skinUrl); + + if (!skinRes.ok) { + return res + .status(404) + .json({ error: "스킨을 찾을 수 없습니다.", url: skinUrl }); + } + + const imageBuffer = await skinRes.arrayBuffer(); + + // S3에 업로드 + await s3Client.send( + new PutObjectCommand({ + Bucket: "minecraft", + Key: s3Key, + Body: Buffer.from(imageBuffer), + ContentType: "image/png", + }) + ); + + console.log(`[Link] 스킨 캐시 완료: ${skinType}/${uuid}/${skinSize}`); + + res.json({ url: s3Url, cached: false }); + } catch (error) { + console.error("[Link] 스킨 캐싱 오류:", error); + // 폴백: mc-heads URL 직접 반환 + res.json({ + url: `https://mc-heads.net/${skinType}/${uuid}/${skinSize}`, + cached: false, + fallback: true, + }); + } +}); + export default router; diff --git a/frontend/src/components/Sidebar.jsx b/frontend/src/components/Sidebar.jsx index a81958e..86bbb58 100644 --- a/frontend/src/components/Sidebar.jsx +++ b/frontend/src/components/Sidebar.jsx @@ -15,7 +15,7 @@ const Sidebar = ({ isMobile = false }) => { const [toast, setToast] = useState(null); const location = useLocation(); const navigate = useNavigate(); - const { isLoggedIn, isAdmin, user, logout } = useAuth(); + const { isLoggedIn, isAdmin, user, logout, checkAuth } = useAuth(); const profileMenuRef = useRef(null); const menuItems = [ @@ -69,16 +69,44 @@ const Sidebar = ({ isMobile = false }) => { fetchLinkStatus(); }, [isLoggedIn, user]); - // 서버 상태 확인 (socket.io) + // 서버 상태 확인 (socket.io) + 닉네임 변경 시에만 동기화 useEffect(() => { const socket = io(window.location.origin, { path: '/socket.io' }); + // 닉네임 동기화 함수 (변경 시에만 호출됨) + const syncNickname = async () => { + if (!isLoggedIn) return; + try { + const token = localStorage.getItem('token'); + const res = await fetch('/link/status', { + headers: { 'Authorization': `Bearer ${token}` } + }); + const data = await res.json(); + if (data.linked) { + setMinecraftLink(data); + // fetch 완료 후 checkAuth로 user.name 갱신 + await checkAuth(); + } + } catch (error) { + // 무시 + } + }; + socket.on('status', (status) => { setServerOnline(status?.online || false); + + // 연동된 유저인 경우, 소켓에서 받은 닉네임과 현재 닉네임 비교 + if (status?.online && minecraftLink?.minecraftUuid && status?.players?.list) { + const playerInGame = status.players.list.find(p => p.uuid === minecraftLink.minecraftUuid); + // 게임 내 닉네임과 현재 저장된 닉네임이 다르면 동기화 + if (playerInGame && playerInGame.name !== user?.name) { + syncNickname(); + } + } }); return () => socket.disconnect(); - }, []); + }, [isLoggedIn, minecraftLink?.minecraftUuid, user?.name]); // 토스트 자동 숨기기 useEffect(() => { @@ -150,7 +178,7 @@ const Sidebar = ({ isMobile = false }) => { className="w-full flex items-center gap-3 px-4 py-2.5 text-sm text-zinc-300 hover:bg-zinc-700 hover:text-white transition-colors" > - 프로필 수정 + 프로필 관리 {minecraftLink && ( diff --git a/frontend/src/pages/Admin.jsx b/frontend/src/pages/Admin.jsx index 618a1f7..d8bf58a 100644 --- a/frontend/src/pages/Admin.jsx +++ b/frontend/src/pages/Admin.jsx @@ -2,31 +2,46 @@ * 관리자 페이지 */ -import { useEffect } from 'react'; -import { useNavigate } from 'react-router-dom'; +import { useEffect, useState } from 'react'; +import { useNavigate, useLocation } from 'react-router-dom'; import { useAuth } from '../contexts/AuthContext'; import { Shield, LogOut, Settings, Server, Users, Loader2 } from 'lucide-react'; +import { motion, AnimatePresence } from 'framer-motion'; export default function Admin() { const { isLoggedIn, isAdmin, user, loading, logout } = useAuth(); const navigate = useNavigate(); + const location = useLocation(); + const [toast, setToast] = useState(null); // 권한 확인 useEffect(() => { if (!loading) { if (!isLoggedIn) { - navigate('/login'); + // 로그인 후 다시 admin으로 돌아올 수 있도록 현재 경로 전달 + navigate('/login', { state: { from: location.pathname } }); } else if (!isAdmin) { - navigate('/'); + setToast('관리자 권한이 필요합니다.'); + setTimeout(() => { + navigate('/'); + }, 1500); } } - }, [isLoggedIn, isAdmin, loading, navigate]); + }, [isLoggedIn, isAdmin, loading, navigate, location.pathname]); const handleLogout = () => { logout(); navigate('/'); }; + // 토스트 자동 숨기기 + useEffect(() => { + if (toast) { + const timer = setTimeout(() => setToast(null), 3000); + return () => clearTimeout(timer); + } + }, [toast]); + if (loading) { return (
@@ -36,7 +51,26 @@ export default function Admin() { } if (!isLoggedIn || !isAdmin) { - return null; + return ( + <> + {/* 토스트 알림 */} + + {toast && ( + + {toast} + + )} + +
+ +
+ + ); } return ( diff --git a/frontend/src/pages/LoginPage.jsx b/frontend/src/pages/LoginPage.jsx index 1199988..34a96ae 100644 --- a/frontend/src/pages/LoginPage.jsx +++ b/frontend/src/pages/LoginPage.jsx @@ -3,17 +3,21 @@ */ import { useState } from 'react'; -import { useNavigate, Link } from 'react-router-dom'; +import { useNavigate, Link, useLocation } from 'react-router-dom'; import { useAuth } from '../contexts/AuthContext'; import { Lock, Mail, AlertCircle, Loader2, Shield, UserPlus } from 'lucide-react'; export default function LoginPage() { const navigate = useNavigate(); + const location = useLocation(); const { login } = useAuth(); const [email, setEmail] = useState(''); const [password, setPassword] = useState(''); const [error, setError] = useState(''); const [loading, setLoading] = useState(false); + + // 로그인 전 접근하려던 경로 (Admin 페이지 등에서 리다이렉트된 경우) + const from = location.state?.from || '/'; const handleSubmit = async (e) => { e.preventDefault(); @@ -25,7 +29,7 @@ export default function LoginPage() { setLoading(false); if (result.success) { - navigate('/admin'); + navigate(from, { replace: true }); } else { setError(result.error); } diff --git a/frontend/src/pages/PlayerStatsPage.jsx b/frontend/src/pages/PlayerStatsPage.jsx index 97bbdb4..1be1fdf 100644 --- a/frontend/src/pages/PlayerStatsPage.jsx +++ b/frontend/src/pages/PlayerStatsPage.jsx @@ -9,15 +9,26 @@ import { formatDate, formatPlayTimeMs } from '../utils/formatters'; // 스티브 3D 스킨 기본 이미지 (로딩 전/실패 시 사용) const STEVE_BODY_BASE64 = ''; -// 플레이어 3D 스킨 컴포넌트 - 로딩 중에는 스티브, 로드 완료 시 실제 스킨 표시 +// 플레이어 3D 스킨 컴포넌트 - 스킨 캐싱 API 사용 const PlayerSkinImage = ({ uuid, playerName }) => { const [src, setSrc] = useState(STEVE_BODY_BASE64); useEffect(() => { - const img = new Image(); - const realUrl = `https://mc-heads.net/body/${uuid}/80`; - img.onload = () => setSrc(realUrl); - img.src = realUrl; + // 스킨 캐싱 API 호출 (body/uuid/size) + fetch(`/link/skin/body/${uuid}/80`) + .then(res => res.json()) + .then(data => { + if (data.url) { + const img = new Image(); + img.onload = () => setSrc(data.url); + img.onerror = () => setSrc(`https://mc-heads.net/body/${uuid}/80`); + img.src = data.url; + } + }) + .catch(() => { + // 폴백: mc-heads 직접 사용 + setSrc(`https://mc-heads.net/body/${uuid}/80`); + }); }, [uuid]); return ( diff --git a/frontend/src/pages/PlayersPage.jsx b/frontend/src/pages/PlayersPage.jsx index 7e6938e..635eae3 100644 --- a/frontend/src/pages/PlayersPage.jsx +++ b/frontend/src/pages/PlayersPage.jsx @@ -8,15 +8,26 @@ import { formatPlayTimeMs } from '../utils/formatters'; // 스티브 머리 기본 이미지 (로딩 전/실패 시 사용) const STEVE_HEAD_BASE64 = ''; -// 플레이어 아바타 컴포넌트 - 로딩 중에는 스티브, 로드 완료 시 실제 스킨 표시 +// 플레이어 아바타 컴포넌트 - 스킨 캐싱 API 사용 const PlayerAvatar = ({ uuid, name }) => { const [src, setSrc] = useState(STEVE_HEAD_BASE64); useEffect(() => { - const img = new Image(); - const realUrl = `https://mc-heads.net/avatar/${uuid}/48`; - img.onload = () => setSrc(realUrl); - img.src = realUrl; + // 스킨 캐싱 API 호출 (avatar/uuid/size) + fetch(`/link/skin/avatar/${uuid}/48`) + .then(res => res.json()) + .then(data => { + if (data.url) { + const img = new Image(); + img.onload = () => setSrc(data.url); + img.onerror = () => setSrc(`https://mc-heads.net/avatar/${uuid}/48`); + img.src = data.url; + } + }) + .catch(() => { + // 폴백: mc-heads 직접 사용 + setSrc(`https://mc-heads.net/avatar/${uuid}/48`); + }); }, [uuid]); return ( diff --git a/frontend/src/pages/ProfilePage.jsx b/frontend/src/pages/ProfilePage.jsx index e7bece3..1a856e0 100644 --- a/frontend/src/pages/ProfilePage.jsx +++ b/frontend/src/pages/ProfilePage.jsx @@ -4,14 +4,16 @@ */ import React, { useState, useEffect } from 'react'; -import { Link, useNavigate } from 'react-router-dom'; +import { Link, useNavigate, useLocation } from 'react-router-dom'; import { useAuth } from '../contexts/AuthContext'; import { ArrowLeft, Copy, Check, Gamepad2, Link as LinkIcon, Unlink, RefreshCw, User, Mail } from 'lucide-react'; import { motion, AnimatePresence } from 'framer-motion'; +import { io } from 'socket.io-client'; export default function ProfilePage() { const navigate = useNavigate(); - const { user, isLoggedIn, loading, checkAuth } = useAuth(); + const location = useLocation(); + const { user, isLoggedIn, loading, checkAuth, logout } = useAuth(); const [linkStatus, setLinkStatus] = useState(null); const [linkToken, setLinkToken] = useState(null); @@ -20,13 +22,14 @@ export default function ProfilePage() { const [isLoading, setIsLoading] = useState(false); const [polling, setPolling] = useState(false); const [showUnlinkDialog, setShowUnlinkDialog] = useState(false); + const [showDeleteDialog, setShowDeleteDialog] = useState(false); // 로그인 체크 (loading 완료 후에만) useEffect(() => { if (!loading && !isLoggedIn) { - navigate('/login'); + navigate('/login', { state: { from: location.pathname } }); } - }, [loading, isLoggedIn, navigate]); + }, [loading, isLoggedIn, navigate, location.pathname]); // 연동 상태 확인 useEffect(() => { @@ -69,6 +72,27 @@ export default function ProfilePage() { return () => window.removeEventListener('beforeunload', handleBeforeUnload); }, [linkToken]); + // 소켓으로 닉네임 변경 감지 시 동기화 + useEffect(() => { + if (!linkStatus?.minecraftUuid) return; + + const socket = io(window.location.origin, { path: '/socket.io' }); + + socket.on('status', async (status) => { + if (status?.online && status?.players?.list) { + const playerInGame = status.players.list.find(p => p.uuid === linkStatus.minecraftUuid); + // 게임 내 닉네임과 현재 저장된 닉네임이 다르면 동기화 + if (playerInGame && playerInGame.name !== user?.name) { + // fetchLinkStatus가 DB 업데이트 → 완료 후 checkAuth로 user.name 갱신 + await fetchLinkStatus(); + await checkAuth(); + } + } + }); + + return () => socket.disconnect(); + }, [linkStatus?.minecraftUuid, user?.name]); + const fetchLinkStatus = async () => { try { const token = localStorage.getItem('token'); @@ -129,6 +153,25 @@ export default function ProfilePage() { } }; + const handleDelete = async () => { + try { + const token = localStorage.getItem('token'); + const res = await fetch('/auth/delete', { + method: 'DELETE', + headers: { 'Authorization': `Bearer ${token}` } + }); + + if (res.ok) { + logout(); + navigate('/'); + } else { + console.error('탈퇴 실패'); + } + } catch (error) { + console.error('탈퇴 실패:', error); + } + }; + if (!user) return null; return ( @@ -265,6 +308,28 @@ export default function ProfilePage() {
)} + + {/* 계정 관리 섹션 */} +
+

+ + 계정 관리 +

+ +
+
+

+ 회원 탈퇴 시 모든 데이터가 삭제되며 복구할 수 없습니다. +

+ +
+
+
{/* 연동 해제 다이얼로그 */} @@ -306,6 +371,47 @@ export default function ProfilePage() { )} + + {/* 탈퇴 확인 다이얼로그 */} + + {showDeleteDialog && ( + <> + setShowDeleteDialog(false)} + /> + +

⚠️ 회원 탈퇴

+

+ 정말 탈퇴하시겠습니까? 모든 데이터가 삭제되며 복구할 수 없습니다. +

+
+ + +
+
+ + )} +
); } + diff --git a/frontend/src/pages/ServerDetail.jsx b/frontend/src/pages/ServerDetail.jsx index cf9d558..b85d971 100644 --- a/frontend/src/pages/ServerDetail.jsx +++ b/frontend/src/pages/ServerDetail.jsx @@ -7,6 +7,34 @@ import Tooltip from '../components/Tooltip'; import { io } from 'socket.io-client'; +// 캐시된 스킨 컴포넌트 (helm 타입 지원) +const CachedSkin = ({ uuid, name, size = 24 }) => { + const [src, setSrc] = useState(null); + const fallbackUrl = `https://mc-heads.net/avatar/${uuid}/${size}`; + + useEffect(() => { + fetch(`/link/skin/avatar/${uuid}/${size}`) + .then(res => res.json()) + .then(data => { + if (data.url) { + setSrc(data.url); + } else { + setSrc(fallbackUrl); + } + }) + .catch(() => setSrc(fallbackUrl)); + }, [uuid, size, fallbackUrl]); + + return ( + {name} { e.target.src = fallbackUrl; }} + /> + ); +}; + const ServerDetail = ({ isMobile = false }) => { const [server, setServer] = useState(null); const [loading, setLoading] = useState(true); @@ -45,7 +73,7 @@ const ServerDetail = ({ isMobile = false }) => { }); socket.on('connect', () => { - console.log('서버 소켓 연결 성공'); + // 소켓 연결 성공 }); socket.on('status', (data) => { @@ -53,7 +81,7 @@ const ServerDetail = ({ isMobile = false }) => { }); socket.on('disconnect', () => { - console.log('서버 소켓 연결 해제'); + // 소켓 연결 해제 }); return () => { @@ -246,15 +274,7 @@ const ServerDetail = ({ isMobile = false }) => { className="flex items-center gap-2.5 px-4 py-2.5 rounded-lg bg-white/5 border border-white/10 hover:border-mc-green/50 hover:bg-mc-green/5 transition-all cursor-pointer group hover:scale-[1.03] active:scale-[0.98]" >
- {player.name} { - e.target.onerror = null; - e.target.src = 'https://minotar.net/helm/Steve/24.png'; - }} - /> + {/* 온라인 표시 */}
diff --git a/frontend/src/pages/WorldsPage.jsx b/frontend/src/pages/WorldsPage.jsx index 100d3a0..1c7a0b1 100644 --- a/frontend/src/pages/WorldsPage.jsx +++ b/frontend/src/pages/WorldsPage.jsx @@ -3,6 +3,34 @@ import { Globe, Sun, CloudRain, CloudLightning, Clock, Users, MapPin, ServerOff import { motion } from 'framer-motion'; import { io } from 'socket.io-client'; +// 캐시된 스킨 컴포넌트 +const CachedSkin = ({ uuid, name, type = 'avatar', size = 40, className }) => { + const [src, setSrc] = useState(null); + const fallbackUrl = `https://mc-heads.net/${type}/${uuid}/${size}`; + + useEffect(() => { + fetch(`/link/skin/${type}/${uuid}/${size}`) + .then(res => res.json()) + .then(data => { + if (data.url) { + setSrc(data.url); + } else { + setSrc(fallbackUrl); + } + }) + .catch(() => setSrc(fallbackUrl)); + }, [uuid, type, size, fallbackUrl]); + + return ( + {name} { e.target.src = fallbackUrl; }} + /> + ); +}; + // 월드 정보 페이지 const WorldsPage = ({ isMobile = false }) => { const [worlds, setWorlds] = useState([]); @@ -188,12 +216,11 @@ const WorldsPage = ({ isMobile = false }) => { className={`flex items-center gap-3 bg-white/5 rounded-xl ${isMobile ? 'p-2' : 'p-3'}`} > {/* 모바일: 머리만, PC: 전신 */} - {player.name}